Source code for rosdistro.vcs
# Software License Agreement (BSD License)
#
# Copyright (c) 2013, Open Source Robotics Foundation, Inc.
# Copyright (c) 2016, Clearpath Robotics
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions
# are met:
#
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above
# copyright notice, this list of conditions and the following
# disclaimer in the documentation and/or other materials provided
# with the distribution.
# * Neither the name of Open Source Robotics Foundation, Inc. nor
# the names of its contributors may be used to endorse or promote
# products derived from this software without specific prior
# written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
# COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
# BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
import os
import re
import subprocess
from distutils.version import LooseVersion
[docs]class Git(object):
_client_executable = None
_client_version = None
def __init__(self, cwd=None):
self.cwd = cwd
if not self._client_executable:
self.__class__._client_executable = _find_executable('git')
[docs] def command(self, *args):
assert self._client_executable is not None, "'git' not found"
return _run_command((self._client_executable,) + args, self.cwd)
[docs] @classmethod
def version_gte(cls, version):
if not cls._client_version:
result = cls().command('--version')
cls._client_version = result['output'].split()[-1]
return LooseVersion(cls._client_version) >= LooseVersion(version)
[docs]def ref_is_hash(ref):
return re.match('^[0-9a-f]{40}$', ref) is not None
def _run_command(cmd, cwd=None, env=None):
result = {'cmd': ' '.join(cmd), 'cwd': cwd}
try:
proc = subprocess.Popen(cmd, cwd=cwd, stdout=subprocess.PIPE, env=env)
output, _ = proc.communicate()
result['output'] = output.rstrip()
result['returncode'] = proc.returncode
except subprocess.CalledProcessError as e:
result['output'] = e.output
result['returncode'] = e.returncode
if not isinstance(result['output'], str):
result['output'] = result['output'].decode('utf-8')
return result
def _find_executable(file_name):
pathext = ['']
if os.name == 'nt':
# https://docs.microsoft.com/en-us/windows-server/administration/windows-commands/start#remarks
# mimic the behavior how CMD.exe searching for a command without the extension specified.
pathext = pathext + os.getenv('PATHEXT').split(os.path.pathsep)
for path in os.getenv('PATH').split(os.path.pathsep):
for ext in pathext:
file_path = os.path.join(path, file_name + ext)
if os.path.isfile(file_path) and os.access(file_path, os.X_OK):
return file_path
return None