普通文本  |  155行  |  3.75 KB

#!/usr/bin/env python
#
# Copyright (C) 2011-2012 The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#      http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#

import datetime
import os
import re
import sys
import subprocess

try:
    import hashlib
    sha1 = hashlib.sha1
except ImportError, e:
    import sha
    sha1 = sha.sha

def get_repo_revision(repo_dir):
    if not os.path.exists(os.path.join(repo_dir, '.git')):
        return 'Unknown (not git)'

    # Get the HEAD revision
    proc = subprocess.Popen(['git', 'log', '-1', '--format=%H'],
                            stdout=subprocess.PIPE,
                            stderr=subprocess.PIPE,
                            cwd=repo_dir)
    out, err = proc.communicate()
    proc.wait()

    rev_sha1 = out.strip()

    # Working Directory Modified
    proc = subprocess.Popen(['git', 'status'],
                            stdout=subprocess.PIPE,
                            stderr=subprocess.PIPE,
                            cwd=repo_dir)
    out, err = proc.communicate()
    proc.wait()

    if out.find('(working directory clean)') == -1:
      mod = ' modified'
    else:
      mod = ''

    return rev_sha1 + mod + ' (git)'

def compute_sha1(path, global_hasher = None):
    f = open(path, 'rb')
    hasher = sha1()
    while True:
        buf = f.read(512)
        hasher.update(buf)
        if global_hasher:
            global_hasher.update(buf)
        if len(buf) < 512:
            break
    f.close()
    return hasher.hexdigest()

def compute_sha1_list(paths):
    hasher = sha1()
    sha1sums = []
    for path in paths:
        sha1sums.append(compute_sha1(path, hasher))
    return (hasher.hexdigest(), sha1sums)

def quote_str(s):
    result = '"'
    for c in s:
        if c == '\\':
            result += '\\\\'
        elif c == '\r':
            result += '\\r'
        elif c == '\n':
            result += '\\n'
        elif c == '\t':
            result += '\\t'
        elif c == '\"':
            result += '\\"'
        elif c == '\'':
            result += '\\\''
        else:
            result += c
    result += '"'
    return result

def main():
    # Check Argument
    if len(sys.argv) < 2:
        print >> sys.stderr, 'USAGE:', sys.argv[0], '[REPO] [LIBs]'
        sys.exit(1)

    # Record Build Time
    build_time = datetime.datetime.now().strftime('%Y/%m/%d %H:%M:%S')

    # Repository Directory (For build revision)
    repo_dir = sys.argv[1]
    build_rev = get_repo_revision(repo_dir)

    # Compute SHA1
    lib_list = list(set(sys.argv[2:]))
    lib_list.sort()
    build_sha1, sha1sum_list = compute_sha1_list(lib_list)

    # Build file list string
    lib_list_str = ''
    for i, path in enumerate(lib_list):
        lib_list_str += '   %s %s\n' % (sha1sum_list[i], path)

    # Print the automatically generated code
    print """/* Automatically generated file (DON'T MODIFY) */

/* Repository directory: %s */

/* File list:
%s*/

#include "bcc/Config/BuildInfo.h"

using namespace bcc;

const char* BuildInfo::GetBuildTime() {
  return %s;
}

const char *BuildInfo::GetBuildRev() {
  return %s;
}

const char *BuildInfo::GetBuildSourceBlob() {
  return %s;
}

""" % (os.path.abspath(repo_dir),
       lib_list_str,
       quote_str(build_time),
       quote_str(build_rev),
       quote_str(build_sha1))

if __name__ == '__main__':
    main()