#!/usr/bin/env python # -*- coding: utf-8 -*- # ########################## Copyrights and license ############################ # # # Copyright 2013 Vincent Jacques # # # # This file is part of PyGithub. # # http://pygithub.github.io/PyGithub/v1/index.html # # # # PyGithub is free software: you can redistribute it and/or modify it under # # the terms of the GNU Lesser General Public License as published by the Free # # Software Foundation, either version 3 of the License, or (at your option) # # any later version. # # # # PyGithub is distributed in the hope that it will be useful, but WITHOUT ANY # # WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS # # FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more # # details. # # # # You should have received a copy of the GNU Lesser General Public License # # along with PyGithub. If not, see . # # # # ############################################################################## import fnmatch import os import subprocess import itertools eightySharps = "################################################################################" def generateLicenseSection(filename): yield "############################ Copyrights and license ############################" yield "# #" for year, name in sorted(listContributors(filename)): line = "# Copyright " + year + " " + name line += (79 - len(line)) * " " + "#" yield line yield "# #" yield "# This file is part of PyGithub. #" yield "# http://pygithub.github.io/PyGithub/v1/index.html #" yield "# #" yield "# PyGithub is free software: you can redistribute it and/or modify it under #" yield "# the terms of the GNU Lesser General Public License as published by the Free #" yield "# Software Foundation, either version 3 of the License, or (at your option) #" yield "# any later version. #" yield "# #" yield "# PyGithub is distributed in the hope that it will be useful, but WITHOUT ANY #" yield "# WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS #" yield "# FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more #" yield "# details. #" yield "# #" yield "# You should have received a copy of the GNU Lesser General Public License #" yield "# along with PyGithub. If not, see . #" yield "# #" yield "################################################################################" def listContributors(filename): contributors = set() for line in subprocess.check_output(["git", "log", "--format=format:%ad %an <%ae>", "--date=short", "--", filename]).split("\n"): year = line[0:4] name = line[11:] contributors.add((year, name)) return contributors def extractBodyLines(lines): bodyLines = [] seenEndOfHeader = False for line in lines: if len(line) > 0 and line[0] != "#": seenEndOfHeader = True if seenEndOfHeader: bodyLines.append(line) # else: # print "HEAD:", line if line == eightySharps: seenEndOfHeader = True # print "BODY:", "\nBODY: ".join(bodyLines) return bodyLines class PythonHeader: def fix(self, filename, lines): isExecutable = lines[0].startswith("#!") newLines = [] if isExecutable: newLines.append("#!/usr/bin/env python") newLines.append("# -*- coding: utf-8 -*-") newLines.append("") for line in generateLicenseSection(filename): newLines.append(line) bodyLines = extractBodyLines(lines) if len(bodyLines) > 0 and bodyLines[0] != "": newLines.append("") if "import " not in bodyLines[0] and bodyLines[0] != '"""' and not bodyLines[0].startswith("##########"): newLines.append("") newLines += bodyLines return newLines class StandardHeader: def fix(self, filename, lines): newLines = [] for line in generateLicenseSection(filename): newLines.append(line) bodyLines = extractBodyLines(lines) if len(bodyLines) and bodyLines[0] != "" > 0: newLines.append("") newLines += bodyLines return newLines def findHeadersAndFiles(): for root, dirs, files in os.walk('.', topdown=True): if ".git" in dirs: dirs.remove(".git") if "developer.github.com" in dirs: dirs.remove("developer.github.com") if "build" in dirs: dirs.remove("build") for filename in files: fullname = os.path.join(root, filename) if filename.endswith(".py"): yield (PythonHeader(), fullname) elif filename in ["COPYING", "COPYING.LESSER"]: pass elif filename.endswith(".rst") or filename.endswith(".md"): pass elif filename == ".gitignore": yield (StandardHeader(), fullname) elif "ReplayData" in fullname: pass elif fullname.endswith(".pyc"): pass else: print "Don't know what to do with", filename def main(): for header, filename in findHeadersAndFiles(): print "Analyzing", filename with open(filename) as f: lines = list(line.rstrip() for line in f) newLines = header.fix(filename, lines) if newLines != lines: print " => actually modifying", filename with open(filename, "w") as f: for line in newLines: f.write(line + "\n") if __name__ == "__main__": main()