1#!/usr/bin/env python
2
3# Copyright 2017 The Glslang Authors. All rights reserved.
4#
5# Licensed under the Apache License, Version 2.0 (the "License");
6# you may not use this file except in compliance with the License.
7# You may obtain a copy of the License at
8#
9#     http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS,
13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14# See the License for the specific language governing permissions and
15# limitations under the License.
16
17"""Get source files for Glslang and its dependencies from public repositories.
18"""
19
20from __future__ import print_function
21
22import argparse
23import json
24import distutils.dir_util
25import os.path
26import subprocess
27import sys
28
29KNOWN_GOOD_FILE = 'known_good.json'
30
31SITE_TO_KNOWN_GOOD_FILE = { 'github' : 'known_good.json',
32                            'gitlab' : 'known_good_khr.json' }
33
34# Maps a site name to its hostname.
35SITE_TO_HOST = { 'github' : 'https://github.com/',
36                 'gitlab' : 'git@gitlab.khronos.org:' }
37
38VERBOSE = True
39
40
41def command_output(cmd, directory, fail_ok=False):
42    """Runs a command in a directory and returns its standard output stream.
43
44    Captures the standard error stream.
45
46    Raises a RuntimeError if the command fails to launch or otherwise fails.
47    """
48    if VERBOSE:
49        print('In {d}: {cmd}'.format(d=directory, cmd=cmd))
50    p = subprocess.Popen(cmd,
51                         cwd=directory,
52                         stdout=subprocess.PIPE)
53    (stdout, _) = p.communicate()
54    if p.returncode != 0 and not fail_ok:
55        raise RuntimeError('Failed to run {} in {}'.format(cmd, directory))
56    if VERBOSE:
57        print(stdout)
58    return stdout
59
60
61def command_retval(cmd, directory):
62    """Runs a command in a directory and returns its return value.
63
64    Captures the standard error stream.
65    """
66    p = subprocess.Popen(cmd,
67                         cwd=directory,
68                         stdout=subprocess.PIPE)
69    p.communicate()
70    return p.returncode
71
72
73class GoodCommit(object):
74    """Represents a good commit for a repository."""
75
76    def __init__(self, json):
77        """Initializes this good commit object.
78
79        Args:
80        'json':  A fully populated JSON object describing the commit.
81        """
82        self._json = json
83        self.name = json['name']
84        self.site = json['site']
85        self.subrepo = json['subrepo']
86        self.subdir = json['subdir'] if ('subdir' in json) else '.'
87        self.commit = json['commit']
88
89    def GetUrl(self):
90        """Returns the URL for the repository."""
91        host = SITE_TO_HOST[self.site]
92        return '{host}{subrepo}'.format(
93                    host=host,
94                    subrepo=self.subrepo)
95
96    def AddRemote(self):
97        """Add the remote 'known-good' if it does not exist."""
98        remotes = command_output(['git', 'remote'], self.subdir).splitlines()
99        if b'known-good' not in remotes:
100            command_output(['git', 'remote', 'add', 'known-good', self.GetUrl()], self.subdir)
101
102    def HasCommit(self):
103        """Check if the repository contains the known-good commit."""
104        return 0 == subprocess.call(['git', 'rev-parse', '--verify', '--quiet',
105                                     self.commit + "^{commit}"],
106                                    cwd=self.subdir)
107
108    def Clone(self):
109        distutils.dir_util.mkpath(self.subdir)
110        command_output(['git', 'clone', self.GetUrl(), '.'], self.subdir)
111
112    def Fetch(self):
113        command_output(['git', 'fetch', 'known-good'], self.subdir)
114
115    def Checkout(self):
116        if not os.path.exists(os.path.join(self.subdir,'.git')):
117            self.Clone()
118        self.AddRemote()
119        if not self.HasCommit():
120            self.Fetch()
121        command_output(['git', 'checkout', self.commit], self.subdir)
122
123
124def GetGoodCommits(site):
125    """Returns the latest list of GoodCommit objects."""
126    known_good_file = SITE_TO_KNOWN_GOOD_FILE[site]
127    with open(known_good_file) as known_good:
128        return [GoodCommit(c) for c in json.loads(known_good.read())['commits']]
129
130
131def main():
132    parser = argparse.ArgumentParser(description='Get Glslang source dependencies at a known-good commit')
133    parser.add_argument('--dir', dest='dir', default='.',
134                        help="Set target directory for Glslang source root. Default is \'.\'.")
135    parser.add_argument('--site', dest='site', default='github',
136                        help="Set git server site. Default is github.")
137
138    args = parser.parse_args()
139
140    commits = GetGoodCommits(args.site)
141
142    distutils.dir_util.mkpath(args.dir)
143    print('Change directory to {d}'.format(d=args.dir))
144    os.chdir(args.dir)
145
146    # Create the subdirectories in sorted order so that parent git repositories
147    # are created first.
148    for c in sorted(commits, key=lambda x: x.subdir):
149        print('Get {n}\n'.format(n=c.name))
150        c.Checkout()
151    sys.exit(0)
152
153
154if __name__ == '__main__':
155    main()
156