2014-11-25 15:14:12 +01:00
|
|
|
#!/usr/bin/python
|
2013-08-22 16:10:23 +02:00
|
|
|
|
2015-02-04 10:46:36 +01:00
|
|
|
# Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+
|
2013-08-22 16:10:23 +02:00
|
|
|
|
2013-06-28 16:36:49 +02:00
|
|
|
import sys
|
|
|
|
import logging
|
|
|
|
import optparse
|
|
|
|
import re
|
|
|
|
import os
|
|
|
|
import shutil
|
|
|
|
import time
|
|
|
|
from cStringIO import StringIO
|
|
|
|
|
|
|
|
MAX_LINE_LENGTH = 80
|
|
|
|
|
|
|
|
FILE_TYPE_CONFIG = {
|
|
|
|
'php': {'prefix': ' * ',
|
|
|
|
'firstComment': '/**',
|
|
|
|
'lastComment': ' */',
|
2013-08-22 16:10:23 +02:00
|
|
|
'linesBefore': 0,
|
2013-06-28 16:36:49 +02:00
|
|
|
'linesAfter': 0},
|
|
|
|
'js': {'prefix': ' * ',
|
|
|
|
'firstComment': '/**',
|
|
|
|
'lastComment': ' */',
|
2013-08-22 16:10:23 +02:00
|
|
|
'linesBefore': 0,
|
|
|
|
'linesAfter': 0},
|
|
|
|
'py': {'prefix': '# ',
|
|
|
|
'firstComment': None,
|
|
|
|
'lastComment': None,
|
|
|
|
'linesBefore': 0,
|
2013-10-23 15:10:33 +02:00
|
|
|
'linesAfter': 0},
|
|
|
|
'less': {'prefix': ' * ',
|
|
|
|
'firstComment': '/**',
|
|
|
|
'lastComment': ' */',
|
|
|
|
'linesBefore': 0,
|
|
|
|
'linesAfter': 0}
|
2013-06-28 16:36:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
REPLACE_TOKENS = {
|
|
|
|
'YEAR': time.strftime('%Y')
|
|
|
|
}
|
|
|
|
|
|
|
|
SECTION_MARKER = re.compile(r'\{\{\{ICINGA_LICENSE_HEADER\}\}\}')
|
|
|
|
|
|
|
|
LICENSE_DATA = None
|
|
|
|
|
|
|
|
__version__ = '1.0'
|
|
|
|
|
|
|
|
__LICENSE_STORE = {}
|
|
|
|
|
|
|
|
__SUFFIX_MATCHER = None
|
|
|
|
|
|
|
|
|
|
|
|
class LogFormatter(logging.Formatter):
|
|
|
|
"""Log formatter with color support which is enabled automaticallyby importing it."""
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
logging.Formatter.__init__(self, *args, **kwargs)
|
|
|
|
self._color = sys.stderr.isatty()
|
|
|
|
if self._color:
|
|
|
|
self._colors = {
|
|
|
|
logging.DEBUG: ('\x1b[34m',), # Blue
|
|
|
|
logging.INFO: ('\x1b[32m',), # Green
|
|
|
|
logging.WARNING: ('\x1b[33m',), # Yellow
|
|
|
|
logging.ERROR: ('\x1b[31m',), # Red
|
|
|
|
logging.CRITICAL: ('\x1b[1m', '\x1b[31m'), # Bold, Red
|
|
|
|
}
|
|
|
|
self._footer = '\x1b[0m'
|
|
|
|
|
|
|
|
def format(self, record):
|
|
|
|
formatted_message = logging.Formatter.format(self, record)
|
|
|
|
if self._color:
|
|
|
|
formatted_message = (''.join(self._colors.get(record.levelno)) + formatted_message +
|
|
|
|
len(self._colors.get(record.levelno)) * self._footer)
|
|
|
|
return formatted_message
|
|
|
|
|
|
|
|
|
|
|
|
def add_optparse_logging_options(parser, default_loglevel='DEBUG'):
|
|
|
|
"""Add log levels to option parser"""
|
|
|
|
LOGLEVELS = ('INFO', 'WARNING', 'ERROR', 'CRITICAL', 'DEBUG')
|
|
|
|
parser.add_option('-v', '--verbose', dest='logging_level', default=default_loglevel, choices=LOGLEVELS,
|
|
|
|
help="Print verbose informational messages. One of %s. [default: %%default]" % ', '.join(
|
|
|
|
LOGLEVELS))
|
|
|
|
|
|
|
|
|
|
|
|
def init_logging(loglevel):
|
|
|
|
"""Initialize loglevels"""
|
|
|
|
channel = logging.StreamHandler()
|
|
|
|
channel.setFormatter(LogFormatter(fmt='%(asctime)-15s: %(message)s', datefmt='%Y-%m-%d %H:%M:%S'))
|
|
|
|
logging.getLogger().addHandler(channel)
|
|
|
|
logging.getLogger().setLevel(getattr(logging, loglevel))
|
|
|
|
return logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
def init_optparse():
|
|
|
|
"""Initialize opt parser"""
|
|
|
|
parser = optparse.OptionParser(usage='%prog -d <DIR> -L <license> -B <suffix>', version='%%prog %s' % __version__)
|
|
|
|
|
|
|
|
parser.add_option("-d", "--directory", action="append", type="string", dest="dir",
|
|
|
|
help="Directory, multiple switches possible")
|
|
|
|
|
|
|
|
parser.add_option("-L", "--license", action="store", type="string", dest="license",
|
|
|
|
help="Path to license file")
|
|
|
|
|
|
|
|
parser.add_option("-B", "--backup", action="store", type="string", dest="backup",
|
|
|
|
help="Backup suffix, e.g. '.BAK'")
|
|
|
|
add_optparse_logging_options(parser)
|
|
|
|
return parser
|
|
|
|
|
|
|
|
|
|
|
|
def match_file_suffix(file_name):
|
|
|
|
"""Test if er have configuration for this file"""
|
|
|
|
global __SUFFIX_MATCHER
|
|
|
|
if __SUFFIX_MATCHER == None:
|
|
|
|
keys = FILE_TYPE_CONFIG.keys()
|
|
|
|
match = r'\.(' + '|'.join(keys) + r')$'
|
|
|
|
__SUFFIX_MATCHER = re.compile(match)
|
|
|
|
return __SUFFIX_MATCHER.search(file_name)
|
|
|
|
|
|
|
|
|
|
|
|
def load_files(dirs):
|
|
|
|
"""Load all files found into an array"""
|
|
|
|
filelist = []
|
|
|
|
for directory in dirs:
|
|
|
|
for root, subFolders, files in os.walk(directory):
|
|
|
|
for file_name in files:
|
|
|
|
if match_file_suffix(file_name):
|
|
|
|
filelist.append(os.path.join(root, file_name))
|
|
|
|
return filelist
|
|
|
|
|
|
|
|
def count_regex_matches(pattern, string):
|
|
|
|
"""Counting regex matchings, e.g. tokens in a file"""
|
|
|
|
total = 0
|
|
|
|
start = 0
|
|
|
|
while True:
|
|
|
|
match_object = pattern.search(string, start)
|
|
|
|
if match_object is None:
|
|
|
|
return total
|
|
|
|
total += 1
|
|
|
|
|
|
|
|
start = match_object.start() + 1
|
|
|
|
|
|
|
|
def test_license_token(data, file_name):
|
|
|
|
"""Test if we have a valid license token in a file"""
|
|
|
|
global SECTION_MARKER
|
|
|
|
c = count_regex_matches(SECTION_MARKER, data)
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
if c == 2:
|
|
|
|
return True
|
|
|
|
elif c == 0:
|
|
|
|
log.warn('No license token in file %s', file_name)
|
|
|
|
elif c < 2:
|
|
|
|
log.error('Incomplete license token in file %s', file_name)
|
|
|
|
else:
|
|
|
|
log.error('More that one license token in file %s', file_name)
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
def get_license(type):
|
|
|
|
"""Creates license data for a specific configuration"""
|
|
|
|
global FILE_TYPE_CONFIG
|
|
|
|
global LICENSE_DATA
|
|
|
|
global REPLACE_TOKENS
|
|
|
|
global __LICENSE_STORE
|
|
|
|
|
|
|
|
try:
|
|
|
|
return __LICENSE_STORE[type]
|
|
|
|
except(KeyError):
|
2014-07-15 12:50:58 +02:00
|
|
|
if not LICENSE_DATA:
|
|
|
|
__LICENSE_STORE[type] = ''
|
|
|
|
return ''
|
2013-06-28 16:36:49 +02:00
|
|
|
config = FILE_TYPE_CONFIG[type]
|
|
|
|
license_data = []
|
2013-08-22 16:10:23 +02:00
|
|
|
license_data.extend([''] * config['linesBefore'])
|
|
|
|
if config['firstComment'] != None:
|
|
|
|
license_data.append(config['firstComment'])
|
2013-06-28 16:36:49 +02:00
|
|
|
for line in LICENSE_DATA.split('\n'):
|
2013-08-22 16:10:23 +02:00
|
|
|
if line:
|
|
|
|
license_data.append(config['prefix'] + line)
|
|
|
|
else:
|
|
|
|
# Whitespace is uselses in this case (#4603)
|
|
|
|
license_data.append(config['prefix'].rstrip())
|
|
|
|
if config['lastComment'] != None:
|
|
|
|
license_data.append(config['lastComment'])
|
2013-06-28 16:36:49 +02:00
|
|
|
license_data.extend([''] * config['linesAfter'])
|
|
|
|
__LICENSE_STORE[type] = '\n'.join(license_data)
|
|
|
|
__LICENSE_STORE[type] = __LICENSE_STORE[type] % REPLACE_TOKENS
|
|
|
|
|
|
|
|
return __LICENSE_STORE[type]
|
|
|
|
|
|
|
|
def read_file_content(file_name):
|
|
|
|
"""Read file into a string"""
|
|
|
|
fhandle = open(file_name, 'r')
|
|
|
|
content = fhandle.read()
|
|
|
|
fhandle.close
|
|
|
|
return content
|
|
|
|
|
|
|
|
def write_file_content(file_name, content):
|
|
|
|
"""Write a string into a file"""
|
|
|
|
fhandle = open(file_name, 'w')
|
|
|
|
fhandle.write(content)
|
|
|
|
fhandle.close()
|
|
|
|
|
|
|
|
def replace_text(org_data, license_data):
|
|
|
|
"""Replace the license token in the string"""
|
|
|
|
shandle = StringIO(org_data)
|
|
|
|
out = ''
|
|
|
|
test = False
|
|
|
|
|
|
|
|
while True:
|
|
|
|
line = shandle.readline()
|
|
|
|
|
|
|
|
if line == '':
|
|
|
|
break
|
|
|
|
|
|
|
|
if SECTION_MARKER.search(line) and test == False:
|
|
|
|
test = True
|
|
|
|
elif SECTION_MARKER.search(line) and test == True:
|
|
|
|
test = False
|
2014-07-15 12:50:58 +02:00
|
|
|
if license_data:
|
|
|
|
out += license_data
|
|
|
|
out += '\n'
|
2013-06-28 16:36:49 +02:00
|
|
|
elif test == True:
|
|
|
|
continue
|
|
|
|
|
|
|
|
out += line
|
|
|
|
shandle.close()
|
|
|
|
return out
|
|
|
|
|
|
|
|
def process_files(files, backup):
|
|
|
|
"""Iterate over files and trigger reokacement"""
|
|
|
|
global FILE_TYPE_CONFIG
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
for file_name in files:
|
|
|
|
data = read_file_content(file_name)
|
|
|
|
if test_license_token(data, file_name):
|
|
|
|
base, ext = os.path.splitext(file_name)
|
|
|
|
ext = ext[1:]
|
|
|
|
try:
|
|
|
|
config = FILE_TYPE_CONFIG[ext]
|
|
|
|
|
|
|
|
license_data = get_license(ext)
|
|
|
|
new_data = replace_text(data, license_data)
|
|
|
|
|
|
|
|
if data != new_data:
|
|
|
|
log.info('File changed: %s', file_name)
|
|
|
|
|
|
|
|
if backup:
|
|
|
|
newfile = file_name + backup
|
|
|
|
log.info('Backup %s to %s', file_name, newfile)
|
|
|
|
shutil.copy(file_name, newfile)
|
|
|
|
|
|
|
|
write_file_content(file_name, new_data)
|
|
|
|
|
|
|
|
log.info('Written to file: %s', file_name)
|
|
|
|
|
|
|
|
except (KeyError):
|
|
|
|
log.error('No header config for file type %s', ext)
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
"""Main script entry point"""
|
|
|
|
global LICENSE_DATA
|
|
|
|
|
|
|
|
parser = init_optparse()
|
|
|
|
(options, args) = parser.parse_args()
|
|
|
|
log = init_logging(options.logging_level)
|
|
|
|
|
|
|
|
if options.dir is None or not len(options.dir) or not options.license:
|
|
|
|
log.error('--dir and --license are mandatory')
|
|
|
|
parser.print_help()
|
|
|
|
return (1)
|
|
|
|
|
|
|
|
log.debug('starting')
|
|
|
|
|
|
|
|
LICENSE_DATA = read_file_content(options.license)
|
|
|
|
|
|
|
|
log.info('Scanning directories ...')
|
|
|
|
files = load_files(options.dir)
|
|
|
|
log.info('Got %d matching ones', len(files))
|
|
|
|
|
|
|
|
log.info('Processing files ...')
|
|
|
|
process_files(files, options.backup)
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2013-10-23 15:10:33 +02:00
|
|
|
sys.exit(main())
|