parent
474803fee4
commit
7a40254bf8
|
@ -1,291 +0,0 @@
|
||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
# Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+
|
|
||||||
|
|
||||||
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': ' */',
|
|
||||||
'linesBefore': 0,
|
|
||||||
'linesAfter': 0},
|
|
||||||
'js': {'prefix': ' * ',
|
|
||||||
'firstComment': '/**',
|
|
||||||
'lastComment': ' */',
|
|
||||||
'linesBefore': 0,
|
|
||||||
'linesAfter': 0},
|
|
||||||
'py': {'prefix': '# ',
|
|
||||||
'firstComment': None,
|
|
||||||
'lastComment': None,
|
|
||||||
'linesBefore': 0,
|
|
||||||
'linesAfter': 0},
|
|
||||||
'less': {'prefix': ' * ',
|
|
||||||
'firstComment': '/**',
|
|
||||||
'lastComment': ' */',
|
|
||||||
'linesBefore': 0,
|
|
||||||
'linesAfter': 0}
|
|
||||||
}
|
|
||||||
|
|
||||||
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):
|
|
||||||
if not LICENSE_DATA:
|
|
||||||
__LICENSE_STORE[type] = ''
|
|
||||||
return ''
|
|
||||||
config = FILE_TYPE_CONFIG[type]
|
|
||||||
license_data = []
|
|
||||||
license_data.extend([''] * config['linesBefore'])
|
|
||||||
if config['firstComment'] != None:
|
|
||||||
license_data.append(config['firstComment'])
|
|
||||||
for line in LICENSE_DATA.split('\n'):
|
|
||||||
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'])
|
|
||||||
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
|
|
||||||
if license_data:
|
|
||||||
out += license_data
|
|
||||||
out += '\n'
|
|
||||||
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__':
|
|
||||||
sys.exit(main())
|
|
Loading…
Reference in New Issue