| # coding: utf-8 |
| from __future__ import unicode_literals, division, absolute_import, print_function |
| |
| import os |
| import threading |
| import sys |
| import shellenv |
| import sublime |
| |
| if sys.version_info < (3,): |
| str_cls = unicode # noqa |
| py2 = True |
| else: |
| str_cls = str |
| py2 = False |
| |
| |
| __version__ = '1.0.0-beta' |
| __version_info__ = (1, 0, 0, 'beta') |
| |
| |
| # The sublime.platform() function will not be available in ST3 upon initial |
| # import, so we determine the platform via the sys.platform value. We cache |
| # the value here to prevent extra IPC calls between plugin_host and |
| # sublime_text in ST3. |
| _platform = { |
| 'win32': 'windows', |
| 'darwin': 'osx' |
| }.get(sys.platform, 'linux') |
| |
| |
| # A special value object to detect if a setting was not found, versus a setting |
| # explicitly being set to null/None in a settings file. We can't use a Python |
| # object here because the value is serialized to json via the ST API. Byte |
| # strings end up turning into an array of integers in ST3. |
| _NO_VALUE = '\x01\x02\x03\x04\x05\x06\x07\x08\x09\x0A\x0B\x0C\x0D\x0E\x0F' |
| |
| |
| class EnvVarError(EnvironmentError): |
| |
| """ |
| An error occurred finding one or more required environment variables |
| """ |
| |
| missing = None |
| |
| |
| class ExecutableError(EnvironmentError): |
| |
| """ |
| An error occurred locating the executable requested |
| """ |
| |
| name = None |
| dirs = None |
| |
| |
| def debug_enabled(): |
| """ |
| Checks to see if the "debug" setting is true |
| |
| :raises: |
| RuntimeError |
| When the function is called from any thread but the UI thread |
| |
| :return: |
| A boolean - if debug is enabled |
| """ |
| |
| # The Sublime Text API is not threadsafe in ST2, so we |
| # double check here to prevent crashes |
| if not isinstance(threading.current_thread(), threading._MainThread): |
| raise RuntimeError('golangconfig.setting_value() must be called from the main thread') |
| |
| value = sublime.load_settings('golang.sublime-settings').get('debug') |
| return False if value == '0' else bool(value) |
| |
| |
| def subprocess_info(executable_name, required_vars, optional_vars=None, view=None, window=None): |
| """ |
| Gathers and formats information necessary to use subprocess.Popen() to |
| run one of the go executables, with details pulled from setting_value() and |
| executable_path(). |
| |
| Ensures that the executable path and env dictionary are properly encoded for |
| Sublime Text 2, where byte strings are necessary. |
| |
| :param executable_name: |
| A unicode string of the executable to locate, e.g. "go" or "gofmt" |
| |
| :param required_vars: |
| A list of unicode strings of the environment variables that are |
| required, e.g. "GOPATH". Obtains values from setting_value(). |
| |
| :param optional_vars: |
| A list of unicode strings of the environment variables that are |
| optional, but should be pulled from setting_value() if available - e.g. |
| "GOOS", "GOARCH". Obtains values from setting_value(). |
| |
| :param view: |
| A sublime.View object to use in finding project-specific settings. This |
| should be passed whenever available. |
| |
| :param window: |
| A sublime.Window object to use in finding project-specific settings. |
| This should be passed whenever available. |
| |
| :raises: |
| RuntimeError |
| When the function is called from any thread but the UI thread |
| TypeError |
| When any of the parameters are of the wrong type |
| golangconfig.ExecutableError |
| When the executable requested could not be located. The .name |
| attribute contains the name of the executable that could not be |
| located. The .dirs attribute contains a list of unicode strings |
| of the directories searched. |
| golangconfig.EnvVarError |
| When one or more required_vars are not available. The .missing |
| attribute will be a list of the names of missing environment |
| variables. |
| |
| :return: |
| A two-element tuple. |
| |
| - [0] A unicode string (byte string for ST2) of the path to the executable |
| - [1] A dict to pass to the env parameter of subprocess.Popen() |
| """ |
| |
| path, _ = executable_path(executable_name, view=view, window=window) |
| if path is None: |
| name = executable_name |
| if sys.platform == 'win32': |
| name += '.exe' |
| dirs = [] |
| settings_path, _ = _get_most_specific_setting('PATH', view=view, window=window) |
| if settings_path and settings_path != _NO_VALUE: |
| dirs.extend(settings_path.split(os.pathsep)) |
| _, shell_dirs = shellenv.get_path() |
| for shell_dir in shell_dirs: |
| if shell_dir not in dirs: |
| dirs.append(shell_dir) |
| exception = ExecutableError( |
| 'The executable "%s" could not be located in any of the following locations: "%s"' % |
| ( |
| name, |
| '", "'.join(dirs) |
| ) |
| ) |
| exception.name = name |
| exception.dirs = dirs |
| raise exception |
| |
| path = shellenv.path_encode(path) |
| |
| _, env = shellenv.get_env(for_subprocess=True) |
| |
| var_groups = [required_vars] |
| if optional_vars: |
| var_groups.append(optional_vars) |
| |
| missing_vars = [] |
| |
| for var_names in var_groups: |
| for var_name in var_names: |
| value, _ = setting_value(var_name, view=view, window=window) |
| var_key = var_name |
| |
| if value is not None: |
| value = str_cls(value) |
| value = shellenv.env_encode(value) |
| var_key = shellenv.env_encode(var_key) |
| |
| if value is None: |
| if var_key in env: |
| del env[var_key] |
| continue |
| |
| env[var_key] = value |
| |
| for required_var in required_vars: |
| var_key = shellenv.env_encode(required_var) |
| if var_key not in env: |
| missing_vars.append(required_var) |
| |
| if missing_vars: |
| missing_vars = sorted(missing_vars, key=lambda s: s.lower()) |
| exception = EnvVarError( |
| 'The following environment variable%s currently unset: %s' % |
| ( |
| 's are' if len(missing_vars) > 1 else ' is', |
| ', '.join(missing_vars) |
| ) |
| ) |
| exception.missing = missing_vars |
| raise exception |
| |
| encoded_goroot = shellenv.env_encode('GOROOT') |
| if encoded_goroot in env: |
| unicode_sep = shellenv.path_decode(os.sep) |
| name = executable_name |
| if sys.platform == 'win32': |
| name += '.exe' |
| relative_executable_path = shellenv.path_encode('bin%s%s' % (unicode_sep, name)) |
| goroot_executable_path = os.path.join(env[encoded_goroot], relative_executable_path) |
| if goroot_executable_path != path: |
| print( |
| 'golangconfig: warning - binary %s was found at "%s", which is not inside of the GOROOT "%s"' % |
| ( |
| executable_name, |
| path, |
| shellenv.path_decode(env[encoded_goroot]) |
| ) |
| ) |
| |
| return (path, env) |
| |
| |
| def setting_value(setting_name, view=None, window=None): |
| """ |
| Returns the user's setting for a specific variable, such as GOPATH or |
| GOROOT. Supports global and per-platform settings. Finds settings by |
| looking in: |
| |
| 1. If a project is open, the project settings |
| 2. The global golang.sublime-settings file |
| 3. The user's environment variables, as defined by their login shell |
| |
| If the setting is a known name, e.g. GOPATH or GOROOT, the value will be |
| checked to ensure the path exists. |
| |
| :param setting_name: |
| A unicode string of the setting to retrieve |
| |
| :param view: |
| A sublime.View object to use in finding project-specific settings. This |
| should be passed whenever available. |
| |
| :param window: |
| A sublime.Window object to use in finding project-specific settings. |
| This should be passed whenever available. |
| |
| :raises: |
| RuntimeError |
| When the function is called from any thread but the UI thread |
| TypeError |
| When any of the parameters are of the wrong type |
| |
| :return: |
| A two-element tuple. |
| |
| If no setting was found, the return value will be: |
| |
| - [0] None |
| - [1] None |
| |
| If a setting was found, the return value will be: |
| |
| - [0] The setting value |
| - [1] The source of the setting, a unicode string: |
| - "project file (os-specific)" |
| - "golang.sublime-settings (os-specific)" |
| - "project file" |
| - "golang.sublime-settings" |
| - A unicode string of the path to the user's login shell |
| |
| The second element of the tuple is intended to be used in the display |
| of debugging information to end users. |
| """ |
| |
| _require_unicode('setting_name', setting_name) |
| _check_view_window(view, window) |
| |
| setting, source = _get_most_specific_setting(setting_name, view, window) |
| |
| if setting == _NO_VALUE: |
| setting = None |
| source = None |
| |
| shell, env = shellenv.get_env() |
| if setting_name in env: |
| source = shell |
| setting = env[setting_name] |
| |
| if setting_name not in set(['GOPATH', 'GOROOT']): |
| return (setting, source) |
| |
| # We add some extra processing here for known settings to improve the |
| # user experience, especially around debugging |
| is_str = isinstance(setting, str_cls) |
| if is_str: |
| if os.path.exists(setting): |
| return (setting, source) |
| |
| if debug_enabled(): |
| print( |
| 'golangconfig: the value for %s from %s - "%s" - does not exist on the filesystem' % |
| ( |
| setting_name, |
| source, |
| setting |
| ) |
| ) |
| |
| elif debug_enabled(): |
| _debug_unicode_string(setting_name, setting, source) |
| |
| return (None, None) |
| |
| |
| def executable_path(executable_name, view=None, window=None): |
| """ |
| Uses the user's Sublime Text settings and then PATH environment variable |
| as set by their login shell to find a go executable |
| |
| :param name: |
| The name of the binary to find - a unicode string of "go", "gofmt" or |
| "godoc" |
| |
| :param view: |
| A sublime.View object to use in finding project-specific settings. This |
| should be passed whenever available. |
| |
| :param window: |
| A sublime.Window object to use in finding project-specific settings. |
| This should be passed whenever available. |
| |
| :raises: |
| RuntimeError |
| When the function is called from any thread but the UI thread |
| TypeError |
| When any of the parameters are of the wrong type |
| |
| :return: |
| A 2-element tuple. |
| |
| If the executable was not found, the return value will be: |
| |
| - [0] None |
| - [1] None |
| |
| If the exeutable was found, the return value will be: |
| |
| - [0] A unicode string of the full path to the executable |
| - [1] A unicode string of the source of the PATH value |
| - "project file (os-specific)" |
| - "golang.sublime-settings (os-specific)" |
| - "project file" |
| - "golang.sublime-settings" |
| - A unicode string of the path to the user's login shell |
| |
| The second element of the tuple is intended to be used in the display |
| of debugging information to end users. |
| """ |
| |
| _require_unicode('executable_name', executable_name) |
| _check_view_window(view, window) |
| |
| executable_suffix = '.exe' if sys.platform == 'win32' else '' |
| suffixed_name = executable_name + executable_suffix |
| |
| setting, source = _get_most_specific_setting('PATH', view, window) |
| if setting is not _NO_VALUE: |
| is_str = isinstance(setting, str_cls) |
| if not is_str: |
| if debug_enabled(): |
| _debug_unicode_string('PATH', setting, source) |
| else: |
| for dir_ in setting.split(os.pathsep): |
| possible_executable_path = os.path.join(dir_, suffixed_name) |
| if _check_executable(possible_executable_path, source, setting): |
| return (possible_executable_path, source) |
| |
| if debug_enabled(): |
| print( |
| 'golangconfig: binary %s not found in PATH from %s - "%s"' % |
| ( |
| executable_name, |
| source, |
| setting |
| ) |
| ) |
| |
| shell, path_dirs = shellenv.get_path() |
| for dir_ in path_dirs: |
| possible_executable_path = os.path.join(dir_, suffixed_name) |
| if _check_executable(possible_executable_path, shell, os.pathsep.join(path_dirs)): |
| return (possible_executable_path, shell) |
| |
| if debug_enabled(): |
| print( |
| 'golangconfig: binary %s not found in PATH from %s - "%s"' % |
| ( |
| executable_name, |
| shell, |
| os.pathsep.join(path_dirs) |
| ) |
| ) |
| |
| return (None, None) |
| |
| |
| def _get_most_specific_setting(name, view, window): |
| """ |
| Looks up a setting in the following order: |
| |
| 1. View settings, looking inside of the "osx", "windows" or "linux" key |
| based on the OS that Sublime Text is running on. These settings are from |
| a project file. |
| 2. Window settings (ST3 only), looking inside of the "osx", "windows" or |
| "linux" key based on the OS that Sublime Text is running on. These |
| settings are from a project file. |
| 3. golang.sublime-settings, looking inside of the "osx", "windows" or |
| "linux" key based on the OS that Sublime Text is running on. |
| 4. The view settings. These settings are from a project file. |
| 5. The window settings (ST3 only). These settings are from a project file. |
| 6. golang.sublime-settings |
| |
| :param name: |
| A unicode string of the setting to fetch |
| |
| :param view: |
| A sublime.View object to use in finding project-specific settings. This |
| should be passed whenever available. |
| |
| :param window: |
| A sublime.Window object to use in finding project-specific settings. |
| This should be passed whenever available. |
| |
| :return: |
| A two-element tuple. |
| |
| If no setting was found, the return value will be: |
| |
| - [0] golangconfig._NO_VALUE |
| - [1] None |
| |
| If a setting was found, the return value will be: |
| |
| - [0] The setting value |
| - [1] A unicode string of the source: |
| - "project file (os-specific)" |
| - "golang.sublime-settings (os-specific)" |
| - "project file" |
| - "golang.sublime-settings" |
| """ |
| |
| # The Sublime Text API is not threadsafe in ST2, so we |
| # double check here to prevent crashes |
| if not isinstance(threading.current_thread(), threading._MainThread): |
| raise RuntimeError('golangconfig.setting_value() must be called from the main thread') |
| |
| if view is not None and not isinstance(view, sublime.View): |
| raise TypeError('view must be an instance of sublime.View, not %s' % _type_name(view)) |
| |
| if window is not None and not isinstance(window, sublime.Window): |
| raise TypeError('window must be an instance of sublime.Window, not %s' % _type_name(window)) |
| |
| st_settings = sublime.load_settings('golang.sublime-settings') |
| |
| view_settings = view.settings().get('golang', {}) if view else {} |
| |
| if view and not window: |
| window = view.window() |
| |
| window_settings = {} |
| if window: |
| if sys.version_info >= (3,): |
| window_settings = window.project_data().get('settings', {}).get('golang', {}) |
| elif not view and window.active_view(): |
| window_settings = window.active_view().settings().get('golang', {}) |
| |
| settings_objects = [ |
| (view_settings, 'project file'), |
| (window_settings, 'project file'), |
| (st_settings, 'golang.sublime-settings'), |
| ] |
| |
| for settings_object, source in settings_objects: |
| platform_settings = settings_object.get(_platform, _NO_VALUE) |
| if platform_settings == _NO_VALUE: |
| continue |
| if not isinstance(platform_settings, dict): |
| continue |
| if platform_settings.get(name, _NO_VALUE) != _NO_VALUE: |
| return (platform_settings.get(name), source + ' (os-specific)') |
| |
| for settings_object, source in settings_objects: |
| result = settings_object.get(name, _NO_VALUE) |
| if result != _NO_VALUE: |
| return (settings_object.get(name), source) |
| |
| return (_NO_VALUE, None) |
| |
| |
| def _require_unicode(name, value): |
| """ |
| Requires that a parameter be a unicode string |
| |
| :param name: |
| A unicode string of the parameter name |
| |
| :param value: |
| The parameter value |
| |
| :raises: |
| TypeError |
| When the value is not a unicode string |
| """ |
| |
| if not isinstance(value, str_cls): |
| raise TypeError('%s must be a unicode string, not %s' % (name, _type_name(value))) |
| |
| |
| def _check_view_window(view, window): |
| """ |
| Ensures that the view and window parameters to a function are suitable |
| objects for our purposes. There is not a strict check for type to allow for |
| mocking during testing. |
| |
| :param view: |
| The view parameter to check |
| |
| :param window: |
| The window parameter to check |
| |
| :raises: |
| TypeError |
| When the view or window parameters are not of the appropriate type |
| """ |
| |
| if view is not None: |
| if not isinstance(view, sublime.View): |
| raise TypeError('view must be an instance of sublime.View, not %s' % _type_name(view)) |
| |
| if window is not None: |
| if not isinstance(window, sublime.Window) and sys.version_info >= (3,): |
| raise TypeError('window must be an instance of sublime.Window, not %s' % _type_name(window)) |
| |
| |
| def _type_name(value): |
| """ |
| :param value: |
| The value to get the type name of |
| |
| :return: |
| A unicode string of the name of the value's type |
| """ |
| |
| value_cls = value.__class__ |
| value_module = value_cls.__module__ |
| if value_module in set(['builtins', '__builtin__']): |
| return value_cls.__name__ |
| |
| return '%s.%s' % (value_module, value_cls.__name__) |
| |
| |
| def _debug_unicode_string(name, value, source): |
| """ |
| Displays a debug message to the console if the value is not a unicode string |
| |
| :param name: |
| A unicode string of the name of the setting |
| |
| :param value: |
| The setting value to check |
| |
| :param source: |
| A unicode string of the source of the setting |
| """ |
| |
| if value is not None and not isinstance(value, str_cls): |
| print( |
| 'golangconfig: the value for %s from %s is not a string, but instead a %s' % |
| ( |
| name, |
| source, |
| _type_name(value) |
| ) |
| ) |
| |
| |
| def _check_executable(possible_executable_path, source, setting): |
| """ |
| Checks to see if a path to an executable exists and that it is, in fact, |
| executable. Will display debug info if the path exists, but is not |
| executable. |
| |
| :param possible_executable_path: |
| A unicode string of the full file path to the executable |
| |
| :param source: |
| A unicode string of the source of the setting |
| |
| :param setting: |
| A unicode string of the PATH value that the executable was found in |
| |
| :return: |
| A boolean - if the possible_executable_path is a file that is executable |
| """ |
| |
| if os.path.exists(possible_executable_path): |
| is_executable = os.path.isfile(possible_executable_path) and os.access(possible_executable_path, os.X_OK) |
| if is_executable: |
| return True |
| |
| if debug_enabled(): |
| executable_name = os.path.basename(possible_executable_path) |
| print( |
| 'golangconfig: binary %s found in PATH from %s - "%s" - is not executable' % |
| ( |
| executable_name, |
| source, |
| setting |
| ) |
| ) |
| |
| return False |