-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsbot_common.py
335 lines (269 loc) · 11 KB
/
sbot_common.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
import sys
import os
import traceback
import collections
import datetime
import pathlib
import shutil
import subprocess
import sublime
import sublime_plugin
# Data type for shared scopes.
HighlightInfo = collections.namedtuple('HighlightInfo', 'scope_name, region_name, type')
# Log levels.
LL_ERROR = 0
LL_INFO = 1
LL_DEBUG = 2
_temp_view_id = None
_mode = int(os.environ.get('SBOT_MODE', 0))
#-----------------------------------------------------------------------------------
#----------------------- Initialization --------------------------------------------
#-----------------------------------------------------------------------------------
# sbot_common is a generic module but needs to know the name of the parent/owner module.
# Get it from the settings file name then initialize known locations for persistence, logging, settings etc.
_friendly_name = ''
_this_dir, _ = os.path.split(__file__)
for fn in os.listdir(_this_dir):
p = pathlib.Path(fn)
if p.suffix == '.sublime-settings':
_friendly_name = p.stem
break
# Now make the useful filenames. Ensure store path exists.
_store_path = os.path.join(sublime.packages_path(), 'User', _friendly_name)
pathlib.Path(_store_path).mkdir(parents=True, exist_ok=True)
_log_fn = os.path.join(_store_path, f'{_friendly_name}.log')
# Initialize logging. Maybe roll over log now.
if os.path.exists(_log_fn) and os.path.getsize(_log_fn) > 50000:
bup = _log_fn.replace('.log', '_old.log')
shutil.copyfile(_log_fn, bup)
# Clear current log file.
with open(_log_fn, 'w'):
pass
#-----------------------------------------------------------------------------------
#---------------------------- Public functions -------------------------------------
#-----------------------------------------------------------------------------------
#-----------------------------------------------------------------------------------
def get_friendly_name():
''' How this is known to humans.'''
return _friendly_name
#-----------------------------------------------------------------------------------
def get_store_fn():
''' Where to keep this module's stuff.'''
return os.path.join(_store_path, f'{_friendly_name}.store')
#-----------------------------------------------------------------------------------
def get_settings_fn():
''' Get the settings fn suitable for ST.'''
return os.path.join(f'{_friendly_name}.sublime-settings')
#-----------------------------------------------------------------------------------
def error(message, tb=None):
'''Logger function.'''
_write_log(LL_ERROR, message, tb)
# Show the user some context info.
info = [message]
# if tb is not None:
# frame = traceback.extract_tb(tb)[-1]
# info.append(f'at {frame.name}({frame.lineno})')
# info.append('See the log for details')
sublime.error_message('\n'.join(info)) # This goes to console too.
#-----------------------------------------------------------------------------------
def info(message):
'''Logger function.'''
_write_log(LL_INFO, message)
sublime.status_message(message)
#-----------------------------------------------------------------------------------
def debug(message):
'''Logger function.'''
if _mode > 0:
_write_log(LL_DEBUG, message)
#-----------------------------------------------------------------------------------
def get_single_caret(view):
'''Get current caret position for one only region. If multiples, return None.'''
if len(view.sel()) == 0:
# raise RuntimeError('No data')
return None
elif len(view.sel()) == 1: # single sel
return view.sel()[0].b
else: # multi sel
return None
#-----------------------------------------------------------------------------------
def get_sel_regions(view):
'''Function to get user selection or the whole view if no selection.'''
regions = []
if len(view.sel()) > 0 and len(view.sel()[0]) > 0: # user sel
regions = view.sel()
else:
regions = [sublime.Region(0, view.size())]
return regions
#-----------------------------------------------------------------------------------
def create_new_view(window, text, reuse=True):
'''Creates or reuse existing temp view with text. Returns the view.'''
view = None
global _temp_view_id
# Locate the current temp view. This will silently fail if there isn't one.
if reuse:
for v in window.views():
if v.id() == _temp_view_id:
view = v
break
if view is None:
# New instance.
view = window.new_file()
view.set_scratch(True)
_temp_view_id = view.id()
# Create/populate the view.
view.run_command('select_all')
view.run_command('cut')
view.run_command('append', {'characters': text}) # insert has some odd behavior - indentation
window.focus_view(view)
return view
#-----------------------------------------------------------------------------------
def wait_load_file(window, fpath, line):
'''Open file asynchronously then position at line. Returns the new View or None if failed.'''
vnew = None
def _load(view):
if view.is_loading():
sublime.set_timeout(lambda: _load(view), 10) # maybe not forever?
else:
view.run_command("goto_line", {"line": line})
# Open the file in a new view.
try:
vnew = window.open_file(fpath)
_load(vnew)
except Exception as e:
error(f'Failed to open {fpath}: {e}', e.__traceback__)
vnew = None
return vnew
#-----------------------------------------------------------------------------------
def get_highlight_info(which='all'):
'''Get list of builtin scope names and corresponding region names as list of HighlightInfo.'''
hl_info = []
if which == 'all' or which == 'user':
for i in range(6): # magic number of markup.user_hl* count.
hl_info.append(HighlightInfo(f'markup.user_hl{i + 1}', f'region_user_hl{i + 1}', 'user'))
if which == 'all' or which == 'fixed':
for i in range(3): # magic number of markup.fixed_hl* count.
hl_info.append(HighlightInfo(f'markup.fixed_hl{i + 1}', f'region_fixed_hl{i + 1}', 'fixed'))
return hl_info
#-----------------------------------------------------------------------------------
def expand_vars(s):
'''Smarter version of builtin. Returns expanded string or None if bad var name.'''
done = False
count = 0
while not done:
if s is not None and '$' in s:
sexp = os.path.expandvars(s)
if s == sexp:
# Invalid var.
s = None
done = True
else:
# Go around again.
s = sexp
else:
# Done expanding.
done = True
# limit iterations
if not done:
count += 1
if count >= 3:
done = True
s = None
return s
#-----------------------------------------------------------------------------------
def get_path_parts(window, paths):
'''
Slide and dice into useful parts. paths is a list of which only the first is considered.
Returns (dir, fn, path) where:
- path is fully expanded path or None if invalid.
- fn is None for a directory.
'''
dir = None
fn = None
path = None
view = window.active_view()
if paths is not None and len(paths) > 0: # came from sidebar
# Get the first element of paths.
path = paths[0]
elif view is not None: # came from view menu
# Get the view file.
path = view.file_name()
else: # maybe image preview - dig out file name
path = window.extract_variables().get('file')
if path is not None:
exp_path = expand_vars(path)
if exp_path is not None:
if os.path.isdir(exp_path):
dir = exp_path
path = exp_path
elif os.path.isfile(exp_path):
path = exp_path
dir, fn = os.path.split(exp_path)
else:
dir = None
fn = None
path = None
return (dir, fn, path)
#-----------------------------------------------------------------------------------
def open_path(path):
'''Acts as if you had clicked the path in the UI. Honors your file associations.'''
if sublime.platform() == 'osx':
subprocess.call(['open', path])
elif sublime.platform() == 'windows':
os.startfile(path)
else: # linux variants
subprocess.run(('xdg-open', path))
return True
#-----------------------------------------------------------------------------------
def open_terminal(where):
'''Open a terminal in where.'''
# TODOF This works for gnome. Maybe should support other desktop types?
# Kde -> konsole
# xfce4 -> xfce4-terminal
# Cinnamon -> x-terminal-emulator
# MATE -> mate-terminal --window
# Unity -> gnome-terminal --profile=Default
if sublime.platform() == 'osx':
os.system(f'open -a Terminal {where}')
elif sublime.platform() == 'windows':
subprocess.run(f'wt -d "{where}"', shell=False, check=False) # W10+
else: # linux
subprocess.run(f'gnome-terminal --working-directory="{where}"', shell=False, check=False)
#-----------------------------------------------------------------------------------
#---------------------------- Private functions ------------------------------------
#-----------------------------------------------------------------------------------
#-----------------------------------------------------------------------------------
def _write_log(level, message, tb=None):
'''Format a standard message with caller info and log it.'''
# if _log_fn == INVALID_FN:
# raise RuntimeError('Logger has not been initialized.')
# Gates. Sometimes get stray empty lines.
if len(message) == 0:
return
if len(message) == 1 and message[0] == '\n':
return
# Get caller info.
frame = sys._getframe(2)
fn = os.path.basename(frame.f_code.co_filename)
line = frame.f_lineno
# f'func = {frame.f_code.co_name}'
# f'mod_name = {frame.f_globals["__name__"]}'
# f'class_name = {frame.f_locals["self"].__class__.__name__}'
slvl = '???'
if level == LL_ERROR: slvl = 'ERR'
elif level == LL_INFO: slvl = 'INF'
elif level == LL_DEBUG: slvl = 'DBG'
time_str = f'{str(datetime.datetime.now())}'[0:-3]
# Write the record. No need to be synchronized across multiple sbot plugins
# as ST docs say that API runs on a single thread.
with open(_log_fn, 'a') as log:
out_line = f'{time_str} {slvl} {fn}:{line} {message}'
log.write(out_line + '\n')
if tb is not None:
# The traceback formatter is a bit ugly - clean it up.
tblines = []
for s in traceback.format_tb(tb):
if len(s) > 0:
tblines.append(s[:-1])
stb = '\n'.join(tblines)
log.write(stb + '\n')
log.flush()