forked from SublimeText/LaTeXTools
-
Notifications
You must be signed in to change notification settings - Fork 0
/
latextools_plugin.py
557 lines (432 loc) · 18.4 KB
/
latextools_plugin.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
'''
Plugin auto-discovery system intended for use in LaTeXTools package.
Overview
========
A plugin is a Python class that extends LaTeXToolsPlugin and provides some
functionality, usually via a function, that LaTeXTools code interacts with.
This module provide mechanisms for loading such plugins from arbitrary files
and configuring the environment in which they are used. It tries to make as
few assumptions as possible about how the consuming code and the plugin
interact. At it's heart it is just a plugin registry.
A quick example plugin:
from latextools_plugin import LaTeXToolsPlugin
class PluginSample(LaTeXToolsPlugin):
def do_something():
pass
And example consuming code:
from latextools_plugin import get_plugin
plugin = get_plugin('plugin_sample')
# instantiate and use the plugin
plugin().do_something()
Note that we make no assumption about how plugins are used, just how they are
loaded. It is up to the consuming code to provide a protocol for interaction,
i.e., methods that will be called, etc.
As shown above, plugin others should import and sub-class the
`LaTeXToolsPlugin` class from this module, as this will ensure the plugin is
properly registered and so available to consuming code. Plugins are registered
by using a version of the class name, so it is important that all plugins used
have a unique class name. The conversion is similar to how Sublime handles
*Command objects, so the name is converted from CamelCase to snake_case. In
addition, the word "Plugin", if it occurs at the end of the class name, is
removed.
Consuming code loads a plugin from the registry by passing the converted plugin
name to the `get_plugin()` function defined in this class. What is returned is
the class itself, i.e., it is the responsibility of the consuming code to
initialize the plugin and then interact with it.
Finding Plugins
===============
The following loading mechanisms for plugins are provided, either using
configuration options or the `add_plugin_path()` function defined in this
module.
Configuration options:
`plugin_paths`: in the standard user configuration.
A list of either directories to search for plugins or paths to
plugins.
Defaults to an empty list, in which case nothing will be done.
Paths can either be specified as absolute paths, paths relative to the
LaTeXTools package or paths relative to the User package. Paths in the
User package will mask paths in the LaTeXTools package. This is
intended to emulate the behaviour of ST.
If the default glob of *.py is unacceptable, the path can instead be
specified as a tuple consisting of the path and the glob to use. The
glob *must* be compatible with the Python glob module. E.g.,
```json
"plugin_paths": [['latextools_plugins', '*.py3']]
```
will load all .py3 files in the `latextools_plugins` subdirectory of
the User package.
API:
`add_plugin_path()`: can be used in a manner similar to the `plugin_paths`
configuration option. Its required argument is the path to be search, which
can be specified either relative to the LaTeXTools package, the User
package or as an absolute path. In addition it takes an optional argument
of the glob to use to identify plugin files. The main purpose is to allow
LaTeXTools code to register a default location to load plugins from.
The Plugin Environment
======================
The plugin environment will be setup so that the directory containing the
plugin is the first entry on sys.path, enabling import of any modules located
in the same folder, according to standard Python import rules. In addition, the
standard modules available to SublimeText are available. In addition, a small
number of modules from LaTeXTools itself can be made available. This list of
modules can be configured either through the `plugins_whitelist` configuration
option in the settings file or by using the `add_whitelist_module()` function
defined in this module.
Configuration options:
`plugins_whitelist`:
A list of LaTeXTools module names to be made available via sys.modules
when loading plugins. This names do not need to be the fully
qualified name, but should be the name of the module relative to the
LaTeXTools folder (i.e. "latextools_utils" rather than
"LaTeXTools.latextools_utils") as this ensures compatibility between
ST2 and ST3.
API:
`add_whitelist_module()`: can be used in a manner similar to the
`plugins_whitelist` option described above, i.e. called with the name of a
module to add to the list of modules available in sys.modules when a
LaTeXTools plugin is loaded. The optional argument, `module`, if used
should be a Python module object (normally obtained from `sys.modules`).
This is primarily intended to expose a module that would not otherwise be
available or expose an already available module to plugins under a
different name.
'''
from __future__ import print_function
import sublime
import glob as _glob
import os
import sys
import threading
import traceback
from contextlib import contextmanager
from collections import MutableMapping
try:
from latextools_utils import get_setting
import latextools_plugin_internal as internal
except ImportError:
from .latextools_utils import get_setting
from . import latextools_plugin_internal as internal
__all__ = [
'LaTeXToolsPlugin', 'get_plugin', 'get_plugins_by_type',
'add_plugin_path', 'LaTeXToolsPluginException', 'InvalidPluginException',
'NoSuchPluginException'
]
# this is used to load plugins and not interfere with other modules
_MODULE_PREFIX = '_latextools_'
# -- Public API --#
# exceptions
class LaTeXToolsPluginException(Exception):
'''
Base class for plugin-related exceptions
'''
pass
class NoSuchPluginException(LaTeXToolsPluginException):
'''
Exception raised if an attempt is made to access a plugin that does not
exist
Intended to allow the consumer to provide the user with some more useful
information e.g., how to properly configure a module for an extension point
'''
pass
class InvalidPluginException(LaTeXToolsPluginException):
'''
Exception raised if an attempt is made to register a plugin that is not a
subclass of LaTeXToolsPlugin.
'''
pass
LaTeXToolsPlugin = internal.LaTeXToolsPlugin
# methods for consumers
def add_plugin_path(path, glob='*.py'):
'''
This function adds plugins from a specified path.
It is primarily intended to be used by consumers to load plugins from a
custom path without needing to access the internals. For example, consuming
code could use this to load any default plugins
`glob`, if specified should be a valid Python glob. See the `glob` module.
'''
if (path, glob) not in internal._REGISTERED_PATHS_TO_LOAD:
internal._REGISTERED_PATHS_TO_LOAD.append((path, glob))
# if we are called before `plugin_loaded`
if internal._REGISTRY is None:
return
previous_plugins = set(internal._REGISTRY.keys())
with _latextools_module_hack():
if not os.path.exists(path):
return
if os.path.isfile(path):
plugin_dir = os.path.dirname(path)
sys.path.insert(0, plugin_dir)
_load_plugin(os.path.basename(path), plugin_dir)
sys.path.pop(0)
else:
for file in _glob.iglob(os.path.join(path, glob)):
plugin_dir = os.path.dirname(file)
sys.path.insert(0, plugin_dir)
_load_plugin(os.path.basename(file), plugin_dir)
sys.path.pop(0)
print('Loaded LaTeXTools plugins {0} from path {1}'.format(
list(set(internal._REGISTRY.keys()) - previous_plugins),
path))
def add_whitelist_module(name, module=None):
'''
API function to ensure that a certain module is made available to any
plugins.
`name` should be the name of the module as it will be imported in a plugin
`module`, if specified, should be either an actual module object or a
callable that returns the actual module object.
The `module` mechanism is provided to allow for the import of modules that
might otherwise be unavailable or available in sys.modules only by a
different name. Standard LaTeXTools modules should provide a name only.
Note that this function *must* be called before add_plugin_path.
'''
for i, (_name, _module) in enumerate(internal._WHITELIST_ADDED):
if _name == name:
if _module == module:
return
internal._WHITELIST_ADDED[i] = (_name, module)
return
internal._WHITELIST_ADDED.append((name, module))
def get_plugin(name):
'''
This is intended to be the main entry-point used by consumers (not
implementors) of plugins, to find any plugins that have registered
themselves by name.
If a plugin cannot be found, a NoSuchPluginException will be thrown. Please
try to provide the user with any helpful information.
Use case:
Provide the user with the ability to load a plugin by a memorable name,
e.g., in a settings file.
For example, 'biblatex' will get the plugin named 'BibLaTeX', etc.
'''
if internal._REGISTRY is None:
raise NoSuchPluginException(
'Could not load plugin {0} because the registry either hasn\'t ' +
'been loaded or has just been unloaded.'.format(name)
)
return internal._REGISTRY[name]
def get_plugins_by_type(cls):
if internal._REGISTRY is None:
raise NoSuchPluginException(
'No plugins could be loaded because the registry either hasn\'t '
'been loaded or has been unloaded'
)
plugins = [plugin for _, plugin in internal._REGISTRY.items()
if issubclass(plugin, cls)]
return plugins
# -- Private API --#
if sys.version_info < (3, 0):
exec("""def reraise(tp, value, tb=None):
raise tp, value, tb
""")
import imp
def _load_module(module_name, filename, *paths):
name, ext = os.path.splitext(filename)
if ext in ('.py', ''):
f, path, description = imp.find_module(name, list(paths))
try:
module = imp.load_module(module_name, f, path, description)
finally:
if f:
f.close()
else:
module = None
exc_info = None
for path in paths:
p = os.path.normpath(os.path.join(path, filename))
if os.path.exists(p):
try:
module = imp.load_source(module_name, filename)
except ImportError:
exc_info = sys.exc_info()
if not module and exc_info:
reraise(*exc_info)
return module
strbase = basestring
FileNotFoundError = IOError
else:
from importlib.machinery import PathFinder, SourceFileLoader
from imp import reload
# WARNING:
# imp module is deprecated in 3.x, unfortunately, importlib does not seem
# to have a stable API, as in 3.4, `find_module` is deprecated in favour of
# `find_spec` and discussions of how best to provide access to the import
# internals seem to be on-going
def _load_module(module_name, filename, *paths):
name, ext = os.path.splitext(filename)
if ext in ('.py', ''):
loader = PathFinder.find_module(name, path=paths)
if loader is None:
loader = PathFinder.find_module(name)
if loader is None:
raise ImportError(
'Could not find module {} on path {} or sys.path'.format(
name, paths))
else:
loader = None
for path in paths:
p = os.path.normpath(os.path.join(path, filename))
if os.path.exists(p):
loader = SourceFileLoader(module_name, p)
if loader is None:
raise ImportError(
'Could not find module {} on path {}'.format(name, paths))
loader.name = module_name
return loader.load_module()
strbase = str
if sublime.version() < '3000':
import latextools_plugin_internal as internal
def _get_sublime_module_name(_, module):
return module
else:
from . import latextools_plugin_internal as internal
def _get_sublime_module_name(directory, module):
return '{0}.{1}'.format(os.path.basename(directory), module)
class LaTeXToolsPluginRegistry(MutableMapping):
'''
Registry used internally to store references to plugins to be retrieved
by plugin consumers.
'''
def __init__(self):
self._registry = {}
def __getitem__(self, key):
try:
return self._registry[key]
except KeyError:
raise NoSuchPluginException(
'Plugin {0} does not exist. Please ensure that the plugin is '
'configured as documented'.format(key)
)
def __setitem__(self, key, value):
if not isinstance(value, internal.LaTeXToolsPluginMeta):
raise InvalidPluginException(value)
self._registry[key] = value
def __delitem__(self, key):
del self._registry[key]
def __iter__(self):
return iter(self._registry)
def __len__(self):
return len(self._registry)
def __str__(self):
return str(self._registry)
_classname_to_internal_name = internal._classname_to_internal_name
def _get_plugin_paths():
plugin_paths = get_setting('plugin_paths', [])
return plugin_paths
def _load_plugin(filename, *paths):
name, ext = os.path.splitext(filename)
# hopefully a unique-enough module name!
if not ext or ext == '.py':
module_name = '{0}{1}'.format(_MODULE_PREFIX, name)
else:
module_name = '{0}{1}_{2}'.format(_MODULE_PREFIX, name, ext[1:])
if module_name in sys.modules:
try:
return sys.modules[module_name]
except ImportError:
# On ST2, this appears to be triggered on the initial reload and
# fails, so instead of reloading just continue to run the loading
# code
pass
except FileNotFoundError:
# A previous plugin has been moved or removed, so just reload it
pass
try:
return _load_module(module_name, filename, *paths)
except:
print('Could not load module {0} using path {1}.'.format(name, paths))
traceback.print_exc()
return None
def _load_plugins():
def _resolve_plugin_path(path):
if not os.path.isabs(path):
p = os.path.normpath(
os.path.join(sublime.packages_path(), 'User', path))
if not os.path.exists(p):
p = os.path.normpath(
os.path.join(sublime.packages_path(), 'LaTeXTools', path))
return p
return path
for path in _get_plugin_paths():
if type(path) == strbase:
add_plugin_path(_resolve_plugin_path(path))
else:
try:
# assume path is a tuple of [<path>, <glob>]
add_plugin_path(_resolve_plugin_path(path[0]), path[1])
except:
print(
'An error occurred while trying to add the plugin '
'path {0}'.format(path))
traceback.print_exc()
@contextmanager
def _latextools_module_hack():
'''
Context manager to ensure sys.modules has certain white-listed modules,
most especially latextools_plugins. This exposes some of the modules in
LaTeXTools to plugins. It is intended primarily to expose library-esque
functionality, such as the getTeXRoot module, but can be configured by
the user as-needed.
'''
# add any white-listed plugins to sys.modules under their own name
plugins_whitelist = get_setting(
'plugins_whitelist',
['external', 'getTeXRoot', 'latextools_utils']
)
# always include latextools_pluing
plugins_whitelist.append('latextools_plugin')
overwritten_modules = {}
whitelist = [(name, None) for name in plugins_whitelist]
whitelist.extend(internal._WHITELIST_ADDED)
# put the directory containing this file on the sys.path
__dir__ = os.path.dirname(__file__)
# handles ST2s relative directory
if __dir__ == '.':
__dir__ = os.path.join(sublime.packages_path(), 'LaTeXTools')
# insert the LaTeXTools directory on the path
sys.path.insert(0, __dir__)
for name, module in whitelist:
if callable(module):
module = module()
if name in sys.modules:
overwritten_modules[name] = sys.modules[name]
# attempting to autoload module
if module is None:
# if the module has already been loaded by ST, we just use that
latextools_module_name = _get_sublime_module_name(__dir__, name)
if latextools_module_name in sys.modules:
sys.modules[name] = sys.modules[latextools_module_name]
else:
try:
sys.modules[name] = _load_module(name, name, __dir__)
except ImportError:
print(
'An error occurred while trying to load white-listed '
'module {0}'.format(name)
)
traceback.print_exc()
else:
sys.modules[name] = module
# remove the LaTeXTools directory from the path
sys.path.pop(0)
yield
# restore any temporarily overwritten modules and clear our loaded modules
for module in plugins_whitelist:
if _get_sublime_module_name(__dir__, module) != module:
del sys.modules[module]
if module in overwritten_modules:
sys.modules[module] = overwritten_modules[module]
# load plugins when the Sublime API is available, just in case...
def plugin_loaded():
t = threading.Thread(target=_plugin_loaded)
t.daemon = True
t.start()
def _plugin_loaded():
internal._REGISTRY = LaTeXToolsPluginRegistry()
print('Loading LaTeXTools plugins...')
for name, cls in internal._REGISTERED_CLASSES_TO_LOAD:
internal._REGISTRY[name] = cls
_load_plugins()
for path, glob in internal._REGISTERED_PATHS_TO_LOAD:
add_plugin_path(path, glob)
# ensure plugin_loaded() called on ST2
if sublime.version() < '3000' and internal._REGISTRY is None:
plugin_loaded()