2013-07-08 05:38:01 +04:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
"""
|
|
|
|
wakatime.projects.git
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
Information about the git project for a given file.
|
|
|
|
|
|
|
|
:copyright: (c) 2013 Alan Hamlett.
|
|
|
|
:license: BSD, see LICENSE for more details.
|
|
|
|
"""
|
|
|
|
|
|
|
|
import logging
|
|
|
|
import os
|
2017-11-06 06:51:43 +03:00
|
|
|
import re
|
2015-08-25 10:42:37 +03:00
|
|
|
import sys
|
2013-07-08 05:38:01 +04:00
|
|
|
|
|
|
|
from .base import BaseProject
|
2014-09-30 20:27:35 +04:00
|
|
|
from ..compat import u, open
|
2013-07-08 05:38:01 +04:00
|
|
|
|
|
|
|
|
2014-07-25 12:01:39 +04:00
|
|
|
log = logging.getLogger('WakaTime')
|
2013-07-08 05:38:01 +04:00
|
|
|
|
|
|
|
|
|
|
|
class Git(BaseProject):
|
2017-11-06 06:51:43 +03:00
|
|
|
_submodule = None
|
|
|
|
_project_name = None
|
|
|
|
_head_file = None
|
2018-09-21 08:29:34 +03:00
|
|
|
_project_folder = None
|
2013-07-08 05:38:01 +04:00
|
|
|
|
2013-07-10 11:14:44 +04:00
|
|
|
def process(self):
|
2017-11-06 06:51:43 +03:00
|
|
|
return self._find_git_config_file(self.path)
|
2013-07-10 11:14:44 +04:00
|
|
|
|
|
|
|
def name(self):
|
2017-11-06 06:51:43 +03:00
|
|
|
return u(self._project_name) if self._project_name else None
|
2013-07-08 05:38:01 +04:00
|
|
|
|
2013-09-07 09:59:03 +04:00
|
|
|
def branch(self):
|
2017-11-06 06:51:43 +03:00
|
|
|
head = self._head_file
|
|
|
|
if head:
|
2018-04-26 18:40:02 +03:00
|
|
|
line = self._first_line_of_file(head)
|
|
|
|
if line is not None:
|
|
|
|
return self._get_branch_from_head_file(line)
|
2016-05-21 15:28:50 +03:00
|
|
|
return u('master')
|
2013-09-07 09:59:03 +04:00
|
|
|
|
2018-09-21 08:29:34 +03:00
|
|
|
def folder(self):
|
|
|
|
return self._project_folder
|
|
|
|
|
2013-12-13 18:35:49 +04:00
|
|
|
def _find_git_config_file(self, path):
|
2013-07-08 05:38:01 +04:00
|
|
|
path = os.path.realpath(path)
|
|
|
|
if os.path.isfile(path):
|
|
|
|
path = os.path.split(path)[0]
|
|
|
|
if os.path.isfile(os.path.join(path, '.git', 'config')):
|
2017-11-06 06:51:43 +03:00
|
|
|
self._project_name = os.path.basename(path)
|
|
|
|
self._head_file = os.path.join(path, '.git', 'HEAD')
|
2018-09-21 08:29:34 +03:00
|
|
|
self._project_folder = path
|
2017-11-06 06:51:43 +03:00
|
|
|
return True
|
2018-04-26 18:40:02 +03:00
|
|
|
|
|
|
|
link_path = self._path_from_gitdir_link_file(path)
|
|
|
|
if link_path:
|
|
|
|
|
|
|
|
# first check if this is a worktree
|
|
|
|
if self._is_worktree(link_path):
|
|
|
|
self._project_name = self._project_from_worktree(link_path)
|
|
|
|
self._head_file = os.path.join(link_path, 'HEAD')
|
2018-09-21 08:29:34 +03:00
|
|
|
self._project_folder = path
|
2018-04-26 18:40:02 +03:00
|
|
|
return True
|
|
|
|
|
|
|
|
# next check if this is a submodule
|
|
|
|
if self._submodules_supported_for_path(path):
|
2017-11-06 06:51:43 +03:00
|
|
|
self._project_name = os.path.basename(path)
|
2018-04-26 18:40:02 +03:00
|
|
|
self._head_file = os.path.join(link_path, 'HEAD')
|
2018-09-21 08:29:34 +03:00
|
|
|
self._project_folder = path
|
2017-11-06 06:51:43 +03:00
|
|
|
return True
|
2018-04-26 18:40:02 +03:00
|
|
|
|
2013-07-08 05:38:01 +04:00
|
|
|
split_path = os.path.split(path)
|
|
|
|
if split_path[1] == '':
|
2017-11-06 06:51:43 +03:00
|
|
|
return False
|
2013-12-13 18:35:49 +04:00
|
|
|
return self._find_git_config_file(split_path[0])
|
2016-06-08 21:43:24 +03:00
|
|
|
|
|
|
|
def _get_branch_from_head_file(self, line):
|
|
|
|
if u(line.strip()).startswith('ref: '):
|
2019-12-05 09:02:51 +03:00
|
|
|
return u(line.strip().split('/', 2)[-1])
|
2016-06-08 21:43:24 +03:00
|
|
|
return None
|
2017-11-06 06:51:43 +03:00
|
|
|
|
|
|
|
def _submodules_supported_for_path(self, path):
|
|
|
|
if not self._configs:
|
|
|
|
return True
|
|
|
|
|
|
|
|
disabled = self._configs.get('submodules_disabled')
|
|
|
|
|
2018-12-19 18:38:18 +03:00
|
|
|
if not disabled or disabled.strip().lower() == 'false':
|
|
|
|
return True
|
2017-11-06 06:51:43 +03:00
|
|
|
if disabled.strip().lower() == 'true':
|
|
|
|
return False
|
|
|
|
|
|
|
|
for pattern in disabled.split("\n"):
|
|
|
|
if pattern.strip():
|
|
|
|
try:
|
|
|
|
compiled = re.compile(pattern, re.IGNORECASE)
|
|
|
|
if compiled.search(path):
|
|
|
|
return False
|
|
|
|
except re.error as ex:
|
|
|
|
log.warning(u('Regex error ({msg}) for disable git submodules pattern: {pattern}').format(
|
|
|
|
msg=u(ex),
|
|
|
|
pattern=u(pattern),
|
|
|
|
))
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
2018-04-26 18:40:02 +03:00
|
|
|
def _is_worktree(self, link_path):
|
|
|
|
return os.path.basename(os.path.dirname(link_path)) == 'worktrees'
|
|
|
|
|
|
|
|
def _path_from_gitdir_link_file(self, path):
|
2017-11-06 06:51:43 +03:00
|
|
|
link = os.path.join(path, '.git')
|
|
|
|
if not os.path.isfile(link):
|
|
|
|
return None
|
|
|
|
|
2018-04-26 18:40:02 +03:00
|
|
|
line = self._first_line_of_file(link)
|
|
|
|
if line is not None:
|
|
|
|
return self._path_from_gitdir_string(path, line)
|
2017-11-06 06:51:43 +03:00
|
|
|
|
|
|
|
return None
|
|
|
|
|
2018-04-26 18:40:02 +03:00
|
|
|
def _path_from_gitdir_string(self, path, line):
|
2017-11-06 06:51:43 +03:00
|
|
|
if line.startswith('gitdir: '):
|
|
|
|
subpath = line[len('gitdir: '):].strip()
|
2018-04-26 18:40:02 +03:00
|
|
|
if os.path.isfile(os.path.join(path, subpath, 'HEAD')):
|
2017-11-23 23:41:59 +03:00
|
|
|
return os.path.realpath(os.path.join(path, subpath))
|
2017-11-06 06:51:43 +03:00
|
|
|
|
|
|
|
return None
|
2018-04-26 18:40:02 +03:00
|
|
|
|
|
|
|
def _project_from_worktree(self, link_path):
|
|
|
|
commondir = os.path.join(link_path, 'commondir')
|
|
|
|
if os.path.isfile(commondir):
|
|
|
|
line = self._first_line_of_file(commondir)
|
|
|
|
if line:
|
|
|
|
gitdir = os.path.abspath(os.path.join(link_path, line))
|
|
|
|
if os.path.basename(gitdir) == '.git':
|
|
|
|
return os.path.basename(os.path.dirname(gitdir))
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
def _first_line_of_file(self, filepath):
|
|
|
|
try:
|
|
|
|
with open(filepath, 'r', encoding='utf-8') as fh:
|
|
|
|
return fh.readline().strip()
|
|
|
|
except UnicodeDecodeError:
|
|
|
|
pass
|
|
|
|
except IOError:
|
|
|
|
pass
|
|
|
|
try:
|
|
|
|
with open(filepath, 'r', encoding=sys.getfilesystemencoding()) as fh:
|
|
|
|
return fh.readline().strip()
|
|
|
|
except:
|
|
|
|
log.traceback(logging.WARNING)
|
|
|
|
|
|
|
|
return None
|