upgrade wakatime-cli to v10.2.4

This commit is contained in:
Alan Hamlett 2018-09-20 22:29:34 -07:00
parent f61a34eda7
commit 360a491cda
13 changed files with 319 additions and 142 deletions

View file

@ -1,7 +1,7 @@
__title__ = 'wakatime' __title__ = 'wakatime'
__description__ = 'Common interface to the WakaTime api.' __description__ = 'Common interface to the WakaTime api.'
__url__ = 'https://github.com/wakatime/wakatime' __url__ = 'https://github.com/wakatime/wakatime'
__version_info__ = ('10', '2', '1') __version_info__ = ('10', '2', '4')
__version__ = '.'.join(__version_info__) __version__ = '.'.join(__version_info__)
__author__ = 'Alan Hamlett' __author__ = 'Alan Hamlett'
__author_email__ = 'alan@wakatime.com' __author_email__ = 'alan@wakatime.com'

View file

@ -20,7 +20,7 @@ import traceback
from .__about__ import __version__ from .__about__ import __version__
from .compat import basestring from .compat import basestring
from .configs import parseConfigFile from .configs import parseConfigFile
from .constants import AUTH_ERROR from .constants import AUTH_ERROR, DEFAULT_SYNC_OFFLINE_ACTIVITY
from .packages import argparse from .packages import argparse
@ -116,6 +116,12 @@ def parse_arguments():
action=StoreWithoutQuotes, action=StoreWithoutQuotes,
help='Optional language name. If valid, takes ' + help='Optional language name. If valid, takes ' +
'priority over auto-detected language.') 'priority over auto-detected language.')
parser.add_argument('--local-file', dest='local_file', metavar='FILE',
action=FileAction,
help='Absolute path to local file for the ' +
'heartbeat. When --entity is a remote file, ' +
'this local file will be used for stats and ' +
'just the value of --entity sent with heartbeat.')
parser.add_argument('--hostname', dest='hostname', parser.add_argument('--hostname', dest='hostname',
action=StoreWithoutQuotes, action=StoreWithoutQuotes,
help='Hostname of current machine.') help='Hostname of current machine.')
@ -126,13 +132,23 @@ def parse_arguments():
parser.add_argument('--disableoffline', dest='offline_deprecated', parser.add_argument('--disableoffline', dest='offline_deprecated',
action='store_true', action='store_true',
help=argparse.SUPPRESS) help=argparse.SUPPRESS)
parser.add_argument('--hide-filenames', dest='hide_filenames', parser.add_argument('--hide-file-names', dest='hide_file_names',
action='store_true', action='store_true',
help='Obfuscate filenames. Will not send file names ' + help='Obfuscate filenames. Will not send file names ' +
'to api.') 'to api.')
parser.add_argument('--hide-filenames', dest='hide_filenames',
action='store_true',
help=argparse.SUPPRESS)
parser.add_argument('--hidefilenames', dest='hidefilenames', parser.add_argument('--hidefilenames', dest='hidefilenames',
action='store_true', action='store_true',
help=argparse.SUPPRESS) help=argparse.SUPPRESS)
parser.add_argument('--hide-project-names', dest='hide_project_names',
action='store_true',
help='Obfuscate project names. When a project ' +
'folder is detected instead of using the ' +
'folder name as the project, a ' +
'.wakatime-project file is created with a ' +
'random project name.')
parser.add_argument('--exclude', dest='exclude', action='append', parser.add_argument('--exclude', dest='exclude', action='append',
help='Filename patterns to exclude from logging. ' + help='Filename patterns to exclude from logging. ' +
'POSIX regex syntax. Can be used more than once.') 'POSIX regex syntax. Can be used more than once.')
@ -170,6 +186,17 @@ def parse_arguments():
action=StoreWithoutQuotes, action=StoreWithoutQuotes,
help='Number of seconds to wait when sending ' + help='Number of seconds to wait when sending ' +
'heartbeats to api. Defaults to 60 seconds.') 'heartbeats to api. Defaults to 60 seconds.')
parser.add_argument('--sync-offline-activity',
dest='sync_offline_activity',
action=StoreWithoutQuotes,
help='Amount of offline activity to sync from your ' +
'local ~/.wakatime.db sqlite3 file to your ' +
'WakaTime Dashboard before exiting. Can be ' +
'"none" or a positive integer number. Defaults ' +
'to 5, meaning for every heartbeat sent while ' +
'online 5 offline heartbeats are synced. Can ' +
'be used without --entity to only sync offline ' +
'activity without generating new heartbeats.')
parser.add_argument('--config', dest='config', action=StoreWithoutQuotes, parser.add_argument('--config', dest='config', action=StoreWithoutQuotes,
help='Defaults to ~/.wakatime.cfg.') help='Defaults to ~/.wakatime.cfg.')
parser.add_argument('--verbose', dest='verbose', action='store_true', parser.add_argument('--verbose', dest='verbose', action='store_true',
@ -214,9 +241,20 @@ def parse_arguments():
if not args.entity: if not args.entity:
if args.file: if args.file:
args.entity = args.file args.entity = args.file
else: elif not args.sync_offline_activity or args.sync_offline_activity == 'none':
parser.error('argument --entity is required') parser.error('argument --entity is required')
if not args.sync_offline_activity:
args.sync_offline_activity = DEFAULT_SYNC_OFFLINE_ACTIVITY
if args.sync_offline_activity == 'none':
args.sync_offline_activity = 0
try:
args.sync_offline_activity = int(args.sync_offline_activity)
if args.sync_offline_activity < 0:
raise Exception('Error')
except:
parser.error('argument --sync-offline-activity must be "none" or an integer number')
if not args.language and args.alternate_language: if not args.language and args.alternate_language:
args.language = args.alternate_language args.language = args.alternate_language
@ -249,24 +287,8 @@ def parse_arguments():
pass pass
if not args.exclude_unknown_project and configs.has_option('settings', 'exclude_unknown_project'): if not args.exclude_unknown_project and configs.has_option('settings', 'exclude_unknown_project'):
args.exclude_unknown_project = configs.getboolean('settings', 'exclude_unknown_project') args.exclude_unknown_project = configs.getboolean('settings', 'exclude_unknown_project')
if not args.hide_filenames and args.hidefilenames: boolean_or_list('hide_file_names', args, configs, alternative_names=['hide_filenames', 'hidefilenames'])
args.hide_filenames = args.hidefilenames boolean_or_list('hide_project_names', args, configs, alternative_names=['hide_projectnames', 'hideprojectnames'])
if args.hide_filenames:
args.hide_filenames = ['.*']
else:
args.hide_filenames = []
option = None
if configs.has_option('settings', 'hidefilenames'):
option = configs.get('settings', 'hidefilenames')
if configs.has_option('settings', 'hide_filenames'):
option = configs.get('settings', 'hide_filenames')
if option is not None:
if option.strip().lower() == 'true':
args.hide_filenames = ['.*']
elif option.strip().lower() != 'false':
for pattern in option.split("\n"):
if pattern.strip() != '':
args.hide_filenames.append(pattern)
if args.offline_deprecated: if args.offline_deprecated:
args.offline = False args.offline = False
if args.offline and configs.has_option('settings', 'offline'): if args.offline and configs.has_option('settings', 'offline'):
@ -307,3 +329,30 @@ def parse_arguments():
print(traceback.format_exc()) print(traceback.format_exc())
return args, configs return args, configs
def boolean_or_list(config_name, args, configs, alternative_names=[]):
"""Get a boolean or list of regexes from args and configs."""
# when argument flag present, set to wildcard regex
for key in alternative_names:
if hasattr(args, key) and getattr(args, key):
setattr(args, config_name, ['.*'])
return
setattr(args, config_name, [])
option = None
alternative_names.insert(0, config_name)
for key in alternative_names:
if configs.has_option('settings', key):
option = configs.get('settings', key)
break
if option is not None:
if option.strip().lower() == 'true':
setattr(args, config_name, ['.*'])
elif option.strip().lower() != 'false':
for pattern in option.split("\n"):
if pattern.strip() != '':
getattr(args, config_name).append(pattern)

View file

@ -45,3 +45,12 @@ Files larger than this in bytes will not have a line count stat for performance.
Default is 2MB. Default is 2MB.
""" """
MAX_FILE_SIZE_SUPPORTED = 2000000 MAX_FILE_SIZE_SUPPORTED = 2000000
""" Default number of offline heartbeats to sync before exiting."""
DEFAULT_SYNC_OFFLINE_ACTIVITY = 100
""" Number of heartbeats per api request.
Even when sending more heartbeats, this is the number of heartbeats sent per
individual https request to the WakaTime API.
"""
HEARTBEATS_PER_REQUEST = 10

View file

@ -42,6 +42,8 @@ class Heartbeat(object):
cursorpos = None cursorpos = None
user_agent = None user_agent = None
_sensitive = ('dependencies', 'lines', 'lineno', 'cursorpos', 'branch')
def __init__(self, data, args, configs, _clone=None): def __init__(self, data, args, configs, _clone=None):
if not data: if not data:
self.skip = u('Skipping because heartbeat data is missing.') self.skip = u('Skipping because heartbeat data is missing.')
@ -83,13 +85,16 @@ class Heartbeat(object):
return return
if self.type == 'file': if self.type == 'file':
self.entity = format_file_path(self.entity) self.entity = format_file_path(self.entity)
if not self.entity or not os.path.isfile(self.entity): if not self._file_exists():
self.skip = u('File does not exist; ignoring this heartbeat.') self.skip = u('File does not exist; ignoring this heartbeat.')
return return
if self._excluded_by_missing_project_file(): if self._excluded_by_missing_project_file():
self.skip = u('Skipping because missing .wakatime-project file in parent path.') self.skip = u('Skipping because missing .wakatime-project file in parent path.')
return return
if args.local_file and not os.path.isfile(args.local_file):
args.local_file = None
project, branch = get_project_info(configs, self, data) project, branch = get_project_info(configs, self, data)
self.project = project self.project = project
self.branch = branch self.branch = branch
@ -104,7 +109,8 @@ class Heartbeat(object):
lineno=data.get('lineno'), lineno=data.get('lineno'),
cursorpos=data.get('cursorpos'), cursorpos=data.get('cursorpos'),
plugin=args.plugin, plugin=args.plugin,
language=data.get('language')) language=data.get('language'),
local_file=args.local_file)
except SkipHeartbeat as ex: except SkipHeartbeat as ex:
self.skip = u(ex) or 'Skipping' self.skip = u(ex) or 'Skipping'
return return
@ -132,7 +138,7 @@ class Heartbeat(object):
Returns a Heartbeat. Returns a Heartbeat.
""" """
if not self.args.hide_filenames: if not self.args.hide_file_names:
return self return self
if self.entity is None: if self.entity is None:
@ -141,29 +147,12 @@ class Heartbeat(object):
if self.type != 'file': if self.type != 'file':
return self return self
for pattern in self.args.hide_filenames: if self.should_obfuscate_filename():
try: self._sanitize_metadata()
compiled = re.compile(pattern, re.IGNORECASE) extension = u(os.path.splitext(self.entity)[1])
if compiled.search(self.entity): self.entity = u('HIDDEN{0}').format(extension)
elif self.should_obfuscate_project():
sanitized = {} self._sanitize_metadata()
sensitive = ['dependencies', 'lines', 'lineno', 'cursorpos', 'branch']
for key, val in self.items():
if key in sensitive:
sanitized[key] = None
else:
sanitized[key] = val
extension = u(os.path.splitext(self.entity)[1])
sanitized['entity'] = u('HIDDEN{0}').format(extension)
return self.update(sanitized)
except re.error as ex:
log.warning(u('Regex error ({msg}) for include pattern: {pattern}').format(
msg=u(ex),
pattern=u(pattern),
))
return self return self
@ -201,6 +190,38 @@ class Heartbeat(object):
is_write=self.is_write, is_write=self.is_write,
) )
def should_obfuscate_filename(self):
"""Returns True if hide_file_names is true or the entity file path
matches one in the list of obfuscated file paths."""
for pattern in self.args.hide_file_names:
try:
compiled = re.compile(pattern, re.IGNORECASE)
if compiled.search(self.entity):
return True
except re.error as ex:
log.warning(u('Regex error ({msg}) for hide_file_names pattern: {pattern}').format(
msg=u(ex),
pattern=u(pattern),
))
return False
def should_obfuscate_project(self):
"""Returns True if hide_project_names is true or the entity file path
matches one in the list of obfuscated project paths."""
for pattern in self.args.hide_project_names:
try:
compiled = re.compile(pattern, re.IGNORECASE)
if compiled.search(self.entity):
return True
except re.error as ex:
log.warning(u('Regex error ({msg}) for hide_project_names pattern: {pattern}').format(
msg=u(ex),
pattern=u(pattern),
))
return False
def _unicode(self, value): def _unicode(self, value):
if value is None: if value is None:
return None return None
@ -211,6 +232,10 @@ class Heartbeat(object):
return None return None
return [self._unicode(value) for value in values] return [self._unicode(value) for value in values]
def _file_exists(self):
return (self.entity and os.path.isfile(self.entity) or
self.args.local_file and os.path.isfile(self.args.local_file))
def _excluded_by_pattern(self): def _excluded_by_pattern(self):
return should_exclude(self.entity, self.args.include, self.args.exclude) return should_exclude(self.entity, self.args.include, self.args.exclude)
@ -224,6 +249,10 @@ class Heartbeat(object):
return False return False
return find_project_file(self.entity) is None return find_project_file(self.entity) is None
def _sanitize_metadata(self):
for key in self._sensitive:
setattr(self, key, None)
def __repr__(self): def __repr__(self):
return self.json() return self.json()

View file

@ -1,81 +1,81 @@
{ {
"actionscript": "ActionScript", "actionscript": "ActionScript",
"apacheconf": "ApacheConf", "apacheconf": "ApacheConf",
"applescript": "AppleScript", "applescript": "AppleScript",
"asp": "ASP", "asp": "ASP",
"assembly": "Assembly", "assembly": "Assembly",
"awk": "Awk", "awk": "Awk",
"bash": "Bash", "bash": "Bash",
"basic": "Basic", "basic": "Basic",
"brightscript": "BrightScript", "brightscript": "BrightScript",
"c": "C", "c": "C",
"c#": "C#", "c#": "C#",
"c++": "C++", "c++": "C++",
"clojure": "Clojure", "clojure": "Clojure",
"cocoa": "Cocoa", "cocoa": "Cocoa",
"coffeescript": "CoffeeScript", "coffeescript": "CoffeeScript",
"coldfusion": "ColdFusion", "coldfusion": "ColdFusion",
"common lisp": "Common Lisp", "common lisp": "Common Lisp",
"cshtml": "CSHTML", "cshtml": "CSHTML",
"css": "CSS", "css": "CSS",
"dart": "Dart", "dart": "Dart",
"delphi": "Delphi", "delphi": "Delphi",
"elixir": "Elixir", "elixir": "Elixir",
"elm": "Elm", "elm": "Elm",
"emacs lisp": "Emacs Lisp", "emacs lisp": "Emacs Lisp",
"erlang": "Erlang", "erlang": "Erlang",
"f#": "F#", "f#": "F#",
"fortran": "Fortran", "fortran": "Fortran",
"go": "Go", "go": "Go",
"gous": "Gosu", "gosu": "Gosu",
"groovy": "Groovy", "groovy": "Groovy",
"haml": "Haml", "haml": "Haml",
"haskell": "Haskell", "haskell": "Haskell",
"haxe": "Haxe", "haxe": "Haxe",
"html": "HTML", "html": "HTML",
"ini": "INI", "ini": "INI",
"jade": "Jade", "jade": "Jade",
"java": "Java", "java": "Java",
"javascript": "JavaScript", "javascript": "JavaScript",
"json": "JSON", "json": "JSON",
"jsx": "JSX", "jsx": "JSX",
"kotlin": "Kotlin", "kotlin": "Kotlin",
"less": "LESS", "less": "LESS",
"lua": "Lua", "lua": "Lua",
"markdown": "Markdown", "markdown": "Markdown",
"matlab": "Matlab", "matlab": "Matlab",
"mustache": "Mustache", "mustache": "Mustache",
"objective-c": "Objective-C", "objective-c": "Objective-C",
"objective-c++": "Objective-C++", "objective-c++": "Objective-C++",
"objective-j": "Objective-J", "objective-j": "Objective-J",
"ocaml": "OCaml", "ocaml": "OCaml",
"perl": "Perl", "perl": "Perl",
"php": "PHP", "php": "PHP",
"powershell": "PowerShell", "powershell": "PowerShell",
"prolog": "Prolog", "prolog": "Prolog",
"puppet": "Puppet", "puppet": "Puppet",
"python": "Python", "python": "Python",
"r": "R", "r": "R",
"restructuredtext": "reStructuredText", "restructuredtext": "reStructuredText",
"ruby": "Ruby", "ruby": "Ruby",
"rust": "Rust", "rust": "Rust",
"sass": "Sass", "sass": "Sass",
"scala": "Scala", "scala": "Scala",
"scheme": "Scheme", "scheme": "Scheme",
"scss": "SCSS", "scss": "SCSS",
"shell": "Shell", "shell": "Shell",
"slim": "Slim", "slim": "Slim",
"smalltalk": "Smalltalk", "smalltalk": "Smalltalk",
"sql": "SQL", "sql": "SQL",
"swift": "Swift", "swift": "Swift",
"text": "Text", "text": "Text",
"turing": "Turing", "turing": "Turing",
"twig": "Twig", "twig": "Twig",
"typescript": "TypeScript", "typescript": "TypeScript",
"typoscript": "TypoScript", "typoscript": "TypoScript",
"vb.net": "VB.net", "vb.net": "VB.net",
"viml": "VimL", "viml": "VimL",
"xaml": "XAML", "xaml": "XAML",
"xml": "XML", "xml": "XML",
"yaml": "YAML" "yaml": "YAML"
} }

View file

@ -24,7 +24,7 @@ from .__about__ import __version__
from .api import send_heartbeats from .api import send_heartbeats
from .arguments import parse_arguments from .arguments import parse_arguments
from .compat import u, json from .compat import u, json
from .constants import SUCCESS, UNKNOWN_ERROR from .constants import SUCCESS, UNKNOWN_ERROR, HEARTBEATS_PER_REQUEST
from .logger import setup_logging from .logger import setup_logging
log = logging.getLogger('WakaTime') log = logging.getLogger('WakaTime')
@ -63,12 +63,22 @@ def execute(argv=None):
msg=u(ex), msg=u(ex),
)) ))
retval = send_heartbeats(heartbeats, args, configs) retval = SUCCESS
while heartbeats:
retval = send_heartbeats(heartbeats[:HEARTBEATS_PER_REQUEST], args, configs)
heartbeats = heartbeats[HEARTBEATS_PER_REQUEST:]
if retval != SUCCESS:
break
if heartbeats:
Queue(args, configs).push_many(heartbeats)
if retval == SUCCESS: if retval == SUCCESS:
queue = Queue(args, configs) queue = Queue(args, configs)
offline_heartbeats = queue.pop_many() for offline_heartbeats in queue.pop_many(args.sync_offline_activity):
if len(offline_heartbeats) > 0:
retval = send_heartbeats(offline_heartbeats, args, configs) retval = send_heartbeats(offline_heartbeats, args, configs)
if retval != SUCCESS:
break
return retval return retval

View file

@ -15,6 +15,7 @@ import os
from time import sleep from time import sleep
from .compat import json from .compat import json
from .constants import DEFAULT_SYNC_OFFLINE_ACTIVITY, HEARTBEATS_PER_REQUEST
from .heartbeat import Heartbeat from .heartbeat import Heartbeat
@ -104,19 +105,23 @@ class Queue(object):
def pop_many(self, limit=None): def pop_many(self, limit=None):
if limit is None: if limit is None:
limit = 5 limit = DEFAULT_SYNC_OFFLINE_ACTIVITY
heartbeats = [] heartbeats = []
count = 0 count = 0
while limit == 0 or count < limit: while count < limit:
heartbeat = self.pop() heartbeat = self.pop()
if not heartbeat: if not heartbeat:
break break
heartbeats.append(heartbeat) heartbeats.append(heartbeat)
count += 1 count += 1
if count % HEARTBEATS_PER_REQUEST == 0:
yield heartbeats
heartbeats = []
return heartbeats if heartbeats:
yield heartbeats
def _get_db_file(self): def _get_db_file(self):
home = '~' home = '~'

View file

@ -9,8 +9,11 @@
:license: BSD, see LICENSE for more details. :license: BSD, see LICENSE for more details.
""" """
import os
import logging import logging
import random
from .compat import open
from .projects.git import Git from .projects.git import Git
from .projects.mercurial import Mercurial from .projects.mercurial import Mercurial
from .projects.projectfile import ProjectFile from .projects.projectfile import ProjectFile
@ -65,6 +68,8 @@ def get_project_info(configs, heartbeat, data):
if project_name is None: if project_name is None:
project_name = data.get('project') or heartbeat.args.project project_name = data.get('project') or heartbeat.args.project
hide_project = heartbeat.should_obfuscate_project()
if project_name is None or branch_name is None: if project_name is None or branch_name is None:
for plugin_cls in REV_CONTROL_PLUGINS: for plugin_cls in REV_CONTROL_PLUGINS:
@ -76,9 +81,18 @@ def get_project_info(configs, heartbeat, data):
if project.process(): if project.process():
project_name = project_name or project.name() project_name = project_name or project.name()
branch_name = branch_name or project.branch() branch_name = branch_name or project.branch()
if hide_project:
branch_name = None
project_name = generate_project_name()
project_file = os.path.join(project.folder(), '.wakatime-project')
try:
with open(project_file, 'w') as fh:
fh.write(project_name)
except IOError:
project_name = None
break break
if project_name is None: if project_name is None and not hide_project:
project_name = data.get('alternate_project') or heartbeat.args.alternate_project project_name = data.get('alternate_project') or heartbeat.args.alternate_project
return project_name, branch_name return project_name, branch_name
@ -88,3 +102,42 @@ def get_configs_for_plugin(plugin_name, configs):
if configs and configs.has_section(plugin_name): if configs and configs.has_section(plugin_name):
return dict(configs.items(plugin_name)) return dict(configs.items(plugin_name))
return None return None
def generate_project_name():
"""Generates a random project name."""
adjectives = [
'aged', 'ancient', 'autumn', 'billowing', 'bitter', 'black', 'blue', 'bold',
'broad', 'broken', 'calm', 'cold', 'cool', 'crimson', 'curly', 'damp',
'dark', 'dawn', 'delicate', 'divine', 'dry', 'empty', 'falling', 'fancy',
'flat', 'floral', 'fragrant', 'frosty', 'gentle', 'green', 'hidden', 'holy',
'icy', 'jolly', 'late', 'lingering', 'little', 'lively', 'long', 'lucky',
'misty', 'morning', 'muddy', 'mute', 'nameless', 'noisy', 'odd', 'old',
'orange', 'patient', 'plain', 'polished', 'proud', 'purple', 'quiet', 'rapid',
'raspy', 'red', 'restless', 'rough', 'round', 'royal', 'shiny', 'shrill',
'shy', 'silent', 'small', 'snowy', 'soft', 'solitary', 'sparkling', 'spring',
'square', 'steep', 'still', 'summer', 'super', 'sweet', 'throbbing', 'tight',
'tiny', 'twilight', 'wandering', 'weathered', 'white', 'wild', 'winter', 'wispy',
'withered', 'yellow', 'young'
]
nouns = [
'art', 'band', 'bar', 'base', 'bird', 'block', 'boat', 'bonus',
'bread', 'breeze', 'brook', 'bush', 'butterfly', 'cake', 'cell', 'cherry',
'cloud', 'credit', 'darkness', 'dawn', 'dew', 'disk', 'dream', 'dust',
'feather', 'field', 'fire', 'firefly', 'flower', 'fog', 'forest', 'frog',
'frost', 'glade', 'glitter', 'grass', 'hall', 'hat', 'haze', 'heart',
'hill', 'king', 'lab', 'lake', 'leaf', 'limit', 'math', 'meadow',
'mode', 'moon', 'morning', 'mountain', 'mouse', 'mud', 'night', 'paper',
'pine', 'poetry', 'pond', 'queen', 'rain', 'recipe', 'resonance', 'rice',
'river', 'salad', 'scene', 'sea', 'shadow', 'shape', 'silence', 'sky',
'smoke', 'snow', 'snowflake', 'sound', 'star', 'sun', 'sun', 'sunset',
'surf', 'term', 'thunder', 'tooth', 'tree', 'truth', 'union', 'unit',
'violet', 'voice', 'water', 'waterfall', 'wave', 'wildflower', 'wind', 'wood'
]
numbers = [str(x) for x in range(10)]
return ' '.join([
random.choice(adjectives).capitalize(),
random.choice(nouns).capitalize(),
random.choice(numbers) + random.choice(numbers),
])

View file

@ -43,3 +43,8 @@ class BaseProject(object):
""" Returns the current branch. """ Returns the current branch.
""" """
raise NotYetImplemented() raise NotYetImplemented()
def folder(self):
""" Returns the project's top folder path.
"""
raise NotYetImplemented()

View file

@ -25,6 +25,7 @@ class Git(BaseProject):
_submodule = None _submodule = None
_project_name = None _project_name = None
_head_file = None _head_file = None
_project_folder = None
def process(self): def process(self):
return self._find_git_config_file(self.path) return self._find_git_config_file(self.path)
@ -40,6 +41,9 @@ class Git(BaseProject):
return self._get_branch_from_head_file(line) return self._get_branch_from_head_file(line)
return u('master') return u('master')
def folder(self):
return self._project_folder
def _find_git_config_file(self, path): def _find_git_config_file(self, path):
path = os.path.realpath(path) path = os.path.realpath(path)
if os.path.isfile(path): if os.path.isfile(path):
@ -47,6 +51,7 @@ class Git(BaseProject):
if os.path.isfile(os.path.join(path, '.git', 'config')): if os.path.isfile(os.path.join(path, '.git', 'config')):
self._project_name = os.path.basename(path) self._project_name = os.path.basename(path)
self._head_file = os.path.join(path, '.git', 'HEAD') self._head_file = os.path.join(path, '.git', 'HEAD')
self._project_folder = path
return True return True
link_path = self._path_from_gitdir_link_file(path) link_path = self._path_from_gitdir_link_file(path)
@ -56,12 +61,14 @@ class Git(BaseProject):
if self._is_worktree(link_path): if self._is_worktree(link_path):
self._project_name = self._project_from_worktree(link_path) self._project_name = self._project_from_worktree(link_path)
self._head_file = os.path.join(link_path, 'HEAD') self._head_file = os.path.join(link_path, 'HEAD')
self._project_folder = path
return True return True
# next check if this is a submodule # next check if this is a submodule
if self._submodules_supported_for_path(path): if self._submodules_supported_for_path(path):
self._project_name = os.path.basename(path) self._project_name = os.path.basename(path)
self._head_file = os.path.join(link_path, 'HEAD') self._head_file = os.path.join(link_path, 'HEAD')
self._project_folder = path
return True return True
split_path = os.path.split(path) split_path = os.path.split(path)

View file

@ -47,6 +47,11 @@ class Mercurial(BaseProject):
log.traceback(logging.WARNING) log.traceback(logging.WARNING)
return u('default') return u('default')
def folder(self):
if self.configDir:
return os.path.dirname(self.configDir)
return None
def _find_hg_config_dir(self, path): def _find_hg_config_dir(self, path):
path = os.path.realpath(path) path = os.path.realpath(path)
if os.path.isfile(path): if os.path.isfile(path):

View file

@ -41,6 +41,11 @@ class Subversion(BaseProject):
return None # pragma: nocover return None # pragma: nocover
return u(self.info['URL'].split('/')[-1].split('\\')[-1]) return u(self.info['URL'].split('/')[-1].split('\\')[-1])
def folder(self):
if 'Repository Root' not in self.info:
return None
return self.info['Repository Root']
def _find_binary(self): def _find_binary(self):
if self.binary_location: if self.binary_location:
return self.binary_location return self.binary_location

View file

@ -40,7 +40,7 @@ log = logging.getLogger('WakaTime')
def get_file_stats(file_name, entity_type='file', lineno=None, cursorpos=None, def get_file_stats(file_name, entity_type='file', lineno=None, cursorpos=None,
plugin=None, language=None): plugin=None, language=None, local_file=None):
if entity_type != 'file': if entity_type != 'file':
stats = { stats = {
'language': None, 'language': None,
@ -52,24 +52,24 @@ def get_file_stats(file_name, entity_type='file', lineno=None, cursorpos=None,
else: else:
language, lexer = standardize_language(language, plugin) language, lexer = standardize_language(language, plugin)
if not language: if not language:
language, lexer = guess_language(file_name) language, lexer = guess_language(file_name, local_file)
language = use_root_language(language, lexer) language = use_root_language(language, lexer)
parser = DependencyParser(file_name, lexer) parser = DependencyParser(local_file or file_name, lexer)
dependencies = parser.parse() dependencies = parser.parse()
stats = { stats = {
'language': language, 'language': language,
'dependencies': dependencies, 'dependencies': dependencies,
'lines': number_lines_in_file(file_name), 'lines': number_lines_in_file(local_file or file_name),
'lineno': lineno, 'lineno': lineno,
'cursorpos': cursorpos, 'cursorpos': cursorpos,
} }
return stats return stats
def guess_language(file_name): def guess_language(file_name, local_file):
"""Guess lexer and language for a file. """Guess lexer and language for a file.
Returns a tuple of (language_str, lexer_obj). Returns a tuple of (language_str, lexer_obj).
@ -81,14 +81,14 @@ def guess_language(file_name):
if language: if language:
lexer = get_lexer(language) lexer = get_lexer(language)
else: else:
lexer = smart_guess_lexer(file_name) lexer = smart_guess_lexer(file_name, local_file)
if lexer: if lexer:
language = u(lexer.name) language = u(lexer.name)
return language, lexer return language, lexer
def smart_guess_lexer(file_name): def smart_guess_lexer(file_name, local_file):
"""Guess Pygments lexer for a file. """Guess Pygments lexer for a file.
Looks for a vim modeline in file contents, then compares the accuracy Looks for a vim modeline in file contents, then compares the accuracy
@ -99,7 +99,7 @@ def smart_guess_lexer(file_name):
text = get_file_head(file_name) text = get_file_head(file_name)
lexer1, accuracy1 = guess_lexer_using_filename(file_name, text) lexer1, accuracy1 = guess_lexer_using_filename(local_file or file_name, text)
lexer2, accuracy2 = guess_lexer_using_modeline(text) lexer2, accuracy2 = guess_lexer_using_modeline(text)
if lexer1: if lexer1: