import configparser import os from deploy.Windows.config import DeployConfig from deploy.Windows.logger import Progress, logger from deploy.Windows.utils import cached_property from deploy.git_over_cdn.client import GitOverCdnClient class GitConfigParser(configparser.ConfigParser): def check(self, section, option, value): result = self.get(section, option, fallback=None) if result == value: logger.info(f'Git config {section}.{option} = {value}') return True else: return False class GitOverCdnClientWindows(GitOverCdnClient): def update(self, *args, **kwargs): Progress.GitInit() _ = super().update(*args, **kwargs) Progress.GitShowVersion() return _ @cached_property def latest_commit(self) -> str: _ = super().latest_commit Progress.GitLatestCommit() return _ def download_pack(self): _ = super().download_pack() Progress.GitDownloadPack() return _ class GitManager(DeployConfig): @staticmethod def remove(file): try: os.remove(file) logger.info(f'Removed file: {file}') except FileNotFoundError: logger.info(f'File not found: {file}') @cached_property def git_config(self): conf = GitConfigParser() conf.read('./.git/config') return conf def git_repository_init( self, repo, source='origin', branch='master', proxy='', ssl_verify=True, keep_changes=False ): logger.hr('Git Init', 1) if not self.execute(f'"{self.git}" init', allow_failure=True): self.remove('./.git/config') self.remove('./.git/index') self.remove('./.git/HEAD') self.remove('./.git/ORIG_HEAD') self.execute(f'"{self.git}" init') Progress.GitInit() logger.hr('Set Git Proxy', 1) if proxy: if not self.git_config.check('http', 'proxy', value=proxy): self.execute(f'"{self.git}" config --local http.proxy {proxy}') if not self.git_config.check('https', 'proxy', value=proxy): self.execute(f'"{self.git}" config --local https.proxy {proxy}') else: if not self.git_config.check('http', 'proxy', value=None): self.execute(f'"{self.git}" config --local --unset http.proxy', allow_failure=True) if not self.git_config.check('https', 'proxy', value=None): self.execute(f'"{self.git}" config --local --unset https.proxy', allow_failure=True) if ssl_verify: if not self.git_config.check('http', 'sslVerify', value='true'): self.execute(f'"{self.git}" config --local http.sslVerify true', allow_failure=True) else: if not self.git_config.check('http', 'sslVerify', value='false'): self.execute(f'"{self.git}" config --local http.sslVerify false', allow_failure=True) Progress.GitSetConfig() logger.hr('Set Git Repository', 1) if not self.git_config.check(f'remote "{source}"', 'url', value=repo): if not self.execute(f'"{self.git}" remote set-url {source} {repo}', allow_failure=True): self.execute(f'"{self.git}" remote add {source} {repo}') Progress.GitSetRepo() logger.hr('Fetch Repository Branch', 1) self.execute(f'"{self.git}" fetch {source} {branch}') Progress.GitFetch() logger.hr('Pull Repository Branch', 1) # Remove git lock for lock_file in [ './.git/index.lock', './.git/HEAD.lock', './.git/refs/heads/master.lock', ]: if os.path.exists(lock_file): logger.info(f'Lock file {lock_file} exists, removing') os.remove(lock_file) if keep_changes: if self.execute(f'"{self.git}" stash', allow_failure=True): self.execute(f'"{self.git}" pull --ff-only {source} {branch}') if self.execute(f'"{self.git}" stash pop', allow_failure=True): pass else: # No local changes to existing files, untracked files not included logger.info('Stash pop failed, there seems to be no local changes, skip instead') else: logger.info('Stash failed, this may be the first installation, drop changes instead') self.execute(f'"{self.git}" reset --hard {source}/{branch}') self.execute(f'"{self.git}" pull --ff-only {source} {branch}') else: self.execute(f'"{self.git}" reset --hard {source}/{branch}') Progress.GitReset() # Since `git fetch` is already called, checkout is faster if not self.execute(f'"{self.git}" checkout {branch}', allow_failure=True): self.execute(f'"{self.git}" pull --ff-only {source} {branch}') Progress.GitCheckout() logger.hr('Show Version', 1) self.execute(f'"{self.git}" --no-pager log --no-merges -1') Progress.GitShowVersion() def git_over_cdn(self): client = GitOverCdnClient( url='https://vip.123pan.cn/1815343254/pack/LmeSzinc_StarRailCopilot_master', folder=self.root_filepath, ) client.logger = logger _ = client.update(keep_changes=self.KeepLocalChanges) return _ def git_install(self): logger.hr('Update Alas', 0) if not self.AutoUpdate: logger.info('AutoUpdate is disabled, skip') Progress.GitShowVersion() return if self.GitOverCdn: if self.git_over_cdn(): return self.git_repository_init( repo=self.Repository, source='origin', branch=self.Branch, proxy=self.GitProxy, ssl_verify=self.SSLVerify, keep_changes=self.KeepLocalChanges, )