-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathpie.py
770 lines (614 loc) · 27.5 KB
/
pie.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
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
from __future__ import print_function
"""
pie - Python Interactive Executor
Enables a user to execute predefined tasks that may accept parameters and options from the command line without any other required packages.
Great for bootstrapping a development environment, and then interacting with it.
"""
__VERSION__='0.3.0i'
import inspect
import os
import re
import shutil
import subprocess
import sys
import traceback
import types
__all__=['task','Parameter','OptionsParameter','options','cmd','cd','env','pip','venv']
# environment constants
WINDOWS=(os.name=='nt')
PY3=(sys.version_info>=(3,0))
# ----------------------------------------
# configuration
# ----------------------------------------
class Lookup(object):
"""
A class that can be used like a dictionary with more succinct syntax:
l=Lookup(name='example',value='good')
print(l.name)
l.newValue=2
"""
def __init__(self,**entries):
self.__dict__.update(entries)
# options is a lookup object where predefined options (in code) can be placed, as well as provided on the command line.
options=Lookup()
# ----------------------------------------
# tasks
# ----------------------------------------
# tasks is a dictionary of registered tasks, where key=name. Tasks are possibly from within submodules where name=module.task.
tasks={}
class TaskWrapper(object):
"""A callable wrapper around a task function. Provides prompting for missing function arguments."""
def __init__(self,fn,params,namespace):
self.fn=fn
self.params=params
self.namespace=namespace
self.desc=fn.__doc__
# parameters set when registering tasks
self.hidden=False
def __call__(self,*args,**kwargs):
# get arg names and defaults from the function
(arg_names,varargs,keywords,defaults)=inspect.getargspec(self.fn)
# map defaults to an arg name
defaults=dict(zip(arg_names[len(arg_names)-len(defaults):],defaults)) if defaults is not None else {}
# map provided values to an arg name
provided=dict(zip(arg_names[:len(args)],args))
provided.update(kwargs)
# for each param, get the value and add to provided if not already there, otherwise make sure it's converted
for param in self.params:
if param.name not in arg_names:
raise Exception('{} not a valid parameter of task {}'.format(param.name,self.fn.__name__))
if param.name not in provided:
# provide a default if one exists
default=defaults.get(param.name,Parameter.NO_VALUE)
provided[param.name]=param.getValue(default)
else:
provided[param.name]=param.convertValue(provided[param.name])
return self.fn(**provided)
def task(parameters=[],namespace=None):
"""
A (function that returns a) decorator that converts a simple Python function into a pie Task.
- parameters is a list of objects (use Lookup) with the following attributes:
name - name of the parameter
conversionFn - a function that will take a string and convert it to the desired type
"""
def decorator(taskFn):
# convert the function into a callable task instance
return TaskWrapper(taskFn,parameters,namespace)
# check in case we were called as a decorator eg. @task (without the function call)
if callable(parameters):
# this means that parameters is actually the function to decorate
taskFn=parameters
# but parameters is used in the wrapper and assumed to be a list, so set it as an empty list (as we weren't provided any parameters)
parameters=[]
return decorator(taskFn)
# otherwise return the decorator function
return decorator
alreadyTraversed=set()
def registerTasksInModule(modulePath,module):
"""Recursively traverse through modules, registering tasks"""
modulePrefix=(modulePath+'.' if modulePath else '')
for k,v in vars(module).items():
if isinstance(v,TaskWrapper):
if k.startswith('_'): v.hidden=True
# handle our various namespace options
k=k.replace('__','.')
mp=modulePrefix+v.namespace+'.'+k if v.namespace is not None else modulePrefix+k
tasks[mp]=v
elif isinstance(v,types.ModuleType):
# there is a situation where a module in a package that is imported into another module (so, doubly nested tasks) will also show up at the package level, even though it's not imported there
# eg. pie_tasks/__init__.py does "from . import submodule", submodule does "from . import doubleNest", but this also causes doubleNest to show up in pie_tasks
# I can't figure out any way to distinguish the difference, but, at the moment, I think it's an obscure enough situation to not worry about yet
# Actually, we have to keep a list of modules we've traversed so that we don't recurse forever (circular imports), this will remove the above problem, but may not fix it correctly...
if v not in alreadyTraversed:
alreadyTraversed.add(v)
registerTasksInModule(modulePrefix+k,v)
def importTasks():
"""Import the pie_tasks module and register all tasks found"""
moduleName=getattr(options,'PIE_TASKS_MODULE','pie_tasks')
m=__import__(moduleName)
registerTasksInModule('',m)
# ----------------------------------------
# parameters to tasks
# ----------------------------------------
class Parameter(object):
"""Parameter base class for specifying how to handle parameters for tasks"""
NO_VALUE=object()
# function for input (also so that we can mock it tests) - TODO: move this to some more generic usable place - InputGetter?
INPUT_FN=input if PY3 else raw_input
def __init__(self,name,prompt=None,inputFn=NO_VALUE,conversionFn=lambda o:o,use_default=False):
"""
name - must match the name of the function argument
prompt - a string that is used if needed to prompt for input
inputFn - the function to use to prompt for input
conversionFn - the function used to convert a string to the correct type
use_default - if True - if the function argument has a default, use it
otherwise, suggest the default in the prompt but still prompt, an empty string input will be converted to the default value
"""
self.name=name
self.prompt=prompt
self.inputFn=inputFn if inputFn is not self.NO_VALUE else self.INPUT_FN
self.conversionFn=conversionFn
self.use_default=use_default
def getValue(self,default):
"""Gets a value for a missing argument, suggesting a default if provided."""
if default is Parameter.NO_VALUE or self.use_default==False:
# prompt for the value
promptStr=self.prompt
if promptStr is None:
defaultStr=' (default: {!r})'.format(default) if default is not Parameter.NO_VALUE else ''
promptStr='Please enter a value for {}{}: '.format(self.name,defaultStr)
v=self.inputFn(promptStr)
if default is not Parameter.NO_VALUE and v=='': v=default
else:
# self.use_default must be True and self.default must not be Parameter.NO_VALUE
v=default
return self.convertValue(v)
def convertValue(self,v):
"""This is separate so that getValue is only used for missing arguments, but convertValue is used for arguments provided on the command line too."""
return self.conversionFn(v)
class OptionsParameter(Parameter):
"""A parameter that is asked for once (or provided on the command line) and then this value is stored in options and used wherever else an OptionsParameter of the same name is referenced."""
def getValue(self,default):
v=getattr(options,self.name,Parameter.NO_VALUE)
if v is Parameter.NO_VALUE:
v=super(OptionsParameter,self).getValue(default)
return v
def convertValue(self,v):
"""Whenever a value is converted, make sure that converted value is persisted to options."""
v=super(OptionsParameter,self).convertValue(v)
setattr(options,self.name,v)
return v
# ----------------------------------------
# operations
# ----------------------------------------
class CmdExecutor(object):
"""
The CmdExecutor (singleton) actually executes a command.
Attributes:
- print_cmd: prints the command that will be executed
- dry_run: does not actually run the command, returns an error code of 0, assuming it passed
- cmd_fun: the function that runs the command
"""
print_cmd=False
dry_run=False
@classmethod
def DEFAULT_CMD_FN(cls,c):
"""The default cmd function using Popen and passing stdout and stderr through"""
if cls.print_cmd:
print(c)
if not cls.dry_run:
p=subprocess.Popen(c,shell=True,stdout=sys.stdout,stderr=sys.stderr,universal_newlines=True)
p.wait()
return p.returncode
return 0
# function to execute a command - can be overridden, must return an error code on failure
cmd_fn=DEFAULT_CMD_FN
class CmdContextManager(object):
"""
The CmdContextManager (singleton) is used to keep track of what context a command is being executed within:
with venv('venv/build'):
cmd('python -m pip')
It also has a class variable `python_cmd` which must be set by any CmdContext that causes the instance of python being used to change.
"""
context=[]
python_cmd=sys.executable
class CmdError(Exception):
def __init__(self,errorcode,cmd):
self.errorcode=errorcode
self.cmd=cmd
@classmethod
def enter(cls,ctx):
cls.context.append(ctx)
return len(cls.context)-1
@classmethod
def cmd(cls,c,i=None):
if i is None: i=len(cls.context)
if i>0: return cls.context[i-1].cmd(c)
errorcode=CmdExecutor.cmd_fn(c)
if errorcode!=0:
raise cls.CmdError(errorcode,c)
return errorcode
@classmethod
def exit(cls):
cls.context.pop()
def cmd(c):
"""Executes a system command (within the current context)"""
return CmdContextManager.cmd(c)
def pip(c,pythonCmd=None):
"""Runs a pip command"""
if pythonCmd is None: pythonCmd=CmdContextManager.python_cmd
cmd('"{}" -m pip {}'.format(pythonCmd,c))
class CmdContext(object):
"""Base class for all cmd context objects."""
# make this a context manager
def __enter__(self):
self.contextPosition=CmdContextManager.enter(self)
self.enter_hook()
return self
def enter_hook(self):
pass
def __exit__(self,exc_type,exc_value,traceback):
self.exit_hook()
CmdContextManager.exit()
# we don't care about an exception
def exit_hook(self):
pass
def cmd(self,c):
"""Execute the cmd within the context"""
return CmdContextManager.cmd(c,self.contextPosition)
class venv(CmdContext):
"""A context class used to execute commands within a virtualenv"""
def __init__(self,path):
self.path=os.path.abspath(path)
def _binary_path(self,binary):
middle='\\Scripts\\' if WINDOWS else '/bin/'
return r'{}{}{}'.format(self.path,middle,binary)
def enter_hook(self):
self.old_python_cmd=CmdContextManager.python_cmd
CmdContextManager.python_cmd=self._binary_path('python')
def exit_hook(self):
CmdContextManager.python_cmd=self.old_python_cmd
def exists(self):
return os.path.isdir(self.path)
def create(self,extraArguments='',pythonCmd=None,py3=PY3):
"""Creates a virutalenv by running the `pythonCmd` and adding `extraArguments` if required. `py3` is used to flag whether this python interpreter is py3 or not. Defaults to whatever the current python version is."""
if pythonCmd is None: pythonCmd=CmdContextManager.python_cmd
venv_module='venv' if py3 else 'virtualenv'
c=r'"{}" -m {} {} "{}"'.format(pythonCmd,venv_module,extraArguments,self.path)
cmd(c)
def is_activated(self):
return self._get_sys_prefix().endswith(self.path)
def pip_update(self):
with self:
pip('install -U pip')
def pip_install_requirements(self,requirements_file='requirements.txt'):
with self:
pip('install -r "{}"'.format(requirements_file))
def cmd(self,c):
"""Runs the command `c` in this virtualenv."""
if WINDOWS:
# cmd.exe /C has real specific behaviour around quotes.
# The below double quote syntax is valid because it strips the outside quotes.
# The path to activate.bat must be quoted in case it contains spaces
c=r'''cmd /c ""{}" && {}"'''.format(self._binary_path('activate.bat'),c)
else:
c=r'bash -c "source "{}" && {}"'.format(self._binary_path('activate'),c)
return CmdContextManager.cmd(c,self.contextPosition)
def destroy(self):
if self.exists():
shutil.rmtree(self.path)
def _get_sys_prefix(self):
if not WINDOWS:
return sys.prefix
# On Windows, running via activate.bat, sys.prefix is converted to short-name format.
# In order to know the sys.prefix path, we need to ensure it's converted back to long name format.
import locale
from ctypes import create_unicode_buffer, FormatError, GetLastError, windll
# Start by getting prefix as unicode (it already is in PY3)
sys_prefix = sys.prefix if PY3 else unicode(sys.prefix)
# long names on Windows (before Windows 10 v1607, without GP changes) require a prefix if longer than MAX_PATH
# just use the prefix everywhere for convenience sake
long_name_prefix = u'\\\\?\\'
sys_prefix = sys_prefix if sys_prefix.startswith(long_name_prefix) else u'{}{}'.format(long_name_prefix, sys_prefix)
# find out how long the long name path is
sys_prefix_chars = windll.kernel32.GetLongPathNameW(sys_prefix, None, 0)
# if we have a char length return, the long name path can be retrieved
if sys_prefix_chars:
# create a buffer based on the char length to hold the long name
sys_prefix_long_name_buffer = create_unicode_buffer(sys_prefix_chars)
# get the long name, inside an if statement to handle the (unlikely) event that the path is deleted
# between the above call and now
if windll.kernel32.GetLongPathNameW(sys_prefix, sys_prefix_long_name_buffer, sys_prefix_chars):
# get the value and remove the prefix
sys_prefix = sys_prefix_long_name_buffer.value
if sys_prefix.startswith(long_name_prefix):
sys_prefix = sys_prefix[len(long_name_prefix):]
return sys_prefix
# check to see if a Windows error was fired
e = GetLastError()
error_template = u'Failed to get long name for sys.prefix ({}): {{}}'.format(sys.prefix)
if e:
formatted_error = FormatError(e).decode(locale.getpreferredencoding(), 'replace')
raise WindowsError(e, error_template.format(formatted_error))
# if no Windows error, who knows what happend, fail
raise Exception(error_template.format('unknown error'))
class cd(CmdContext):
"""A context class used to execute commands within a different working dir"""
def __init__(self,path):
self.path=path
self.old_path=os.getcwd()
def enter_hook(self):
os.chdir(self.path)
def exit_hook(self):
os.chdir(self.old_path)
class env(CmdContext):
"""A context class used to execute commands with a different environment"""
def __init__(self,env_dict):
self.env_dict=env_dict
def enter_hook(self):
self.old_env_dict=self.get_multiple(self.env_dict.keys(),default=None)
self.set_multiple(self.env_dict)
def exit_hook(self):
self.set_multiple(self.old_env_dict)
# some convenience methods
@classmethod
def has(cls,k):
"""Checks if an environment variable exists."""
return (cls.get(k,None) is not None)
@classmethod
def get(cls,k,default=None):
"""Gets a single environment variable. Returns `default` if that variable doesn't exist."""
return os.environ.get(k,default)
@classmethod
def get_multiple(cls,ks,default=None):
"""Accepts a list of keys and returns a dict of values"""
return {k:cls.get(k,default) for k in ks}
@classmethod
def set(cls,k,v):
"""Sets a single environment variable. A value of None results in that environment variable being removed."""
if v is None:
if k in os.environ:
del os.environ[k]
else:
os.environ[k]=v
@classmethod
def set_multiple(cls,d):
"""Accepts a dictionary of values to set"""
for k,v in d.items():
cls.set(k,v)
# ----------------------------------------
# Command line functionality
# ----------------------------------------
class Argument(object):
# a flag to indicate that tasks must be imported to execute this argument
needsTasksImported=False
def execute(self):
raise NotImplemented()
def __repr__(self):
return self.__class__.__name__
class Version(Argument):
def execute(self):
print('pie v{}'.format(__VERSION__))
def __repr__(self):
return 'Version: {}'.format(__VERSION__)
class Verbose(Argument):
def execute(self):
CmdExecutor.print_cmd=True
class DryRun(Argument):
def execute(self):
CmdExecutor.dry_run=True
class CreatePieVenv(Argument):
def execute(self):
pv=PieVenv()
print('Creating {}'.format(pv.PIE_VENV))
pv.create()
class UpdatePieVenv(Argument):
def execute(self):
pv=PieVenv()
print('Updating {} with {}'.format(pv.PIE_VENV,pv.PIE_REQUIREMENTS))
pv.update()
class CreateBatchFile(Argument):
def execute(self):
pythonHome=os.environ.get('PYTHONHOME','')
pythonDir=os.path.dirname(sys.executable)
if WINDOWS:
with open('pie.bat','w') as fout:
fout.write('@echo off\n')
fout.write('set PIE_OLD_PYTHONHOME=%PYTHONHOME%\n')
fout.write('set PYTHONHOME={}\n'.format(pythonHome))
fout.write('set PIE_OLD_PATH=%PATH%\n')
fout.write('set PATH={};%PATH%\n'.format(pythonDir))
fout.write('python -m pie %*\n')
fout.write('set PYTHONHOME=%PIE_OLD_PYTHONHOME%\n')
fout.write('set PIE_OLD_PYTHONHOME=\n')
fout.write('set PATH=%PIE_OLD_PATH%\n')
fout.write('set PIE_OLD_PATH=\n')
else:
with open('pie','w') as fout:
fout.write('export PIE_OLD_PYTHONHOME=$PYTHONHOME\n')
fout.write('export PYTHONHOME={}\n'.format(pythonHome))
fout.write('export PIE_OLD_PATH=$PATH\n')
fout.write('export PATH={}:$PATH\n'.format(pythonDir))
fout.write('python -m pie %*\n')
fout.write('export PYTHONHOME=$PIE_OLD_PYTHONHOME\n')
fout.write('unset PIE_OLD_PYTHONHOME\n')
fout.write('export PATH=$PIE_OLD_PATH\n')
fout.write('unset PIE_OLD_PATH\n')
# TODO: set exec perms
class ListTasks(Argument):
needsTasksImported=True
def __init__(self,includeDescription=True):
self.includeDescription=includeDescription
def execute(self):
for k in sorted(tasks.keys()):
v=tasks[k]
if v.hidden: continue
if self.includeDescription:
desc=v.desc or ''
print('{:30} {:.70}'.format(k,desc.replace('\n',' ')))
else:
print(k)
class Help(Argument):
def execute(self):
print('Usage: pie [ -V | -h | -b | -l | -L | -m <name> | -R | -r | -n | -v ]')
print(' pie [ -o <name>=<value> | <task>[(<args>...)] ]...')
print('Version: v{}'.format(__VERSION__))
print('')
print(' -V Display version')
print(' -h Display this help')
print(' -b Create batch file shortcut')
print(' -l List available tasks with description')
print(' -L List available tasks with name only')
print(' -m <n> Change name of the pie_tasks module to import')
print(' -R Creates a .venv-pie venv using requirements.pie.txt')
print(' -r Updates the .venv-pie venv using requirements.pie.txt')
print(' -n Dry run; don\'t actually execute the commands')
print(' -v Verbose output')
print(' -o Sets an option with name to value')
print(' <task> Runs a task passing through arguments if required')
print('')
print('The order of -o and <task> options matters - each will be executed in the order given on the command line.')
class Option(Argument):
def __init__(self,name,value):
self.name=name
self.value=value
def execute(self):
setattr(options,self.name,self.value)
def __repr__(self):
return 'Option: {}={}'.format(self.name,self.value)
class ModuleName(Argument):
def __init__(self,name):
self.name=name
def execute(self):
setattr(options,'PIE_TASKS_MODULE',self.name)
def __repr__(self):
return 'ModuleName: {}'.format(self.name)
class TaskCall(Argument):
needsTasksImported=True
class TaskNotFound(Exception):
def __init__(self,name):
self.name=name
def __init__(self,name,args=[],kwargs={}):
self.name=name
self.args=args
self.kwargs=kwargs
def execute(self):
if self.name in tasks: tasks[self.name](*self.args,**self.kwargs)
else: raise self.TaskNotFound(self.name)
def __repr__(self):
return 'Task: {}(args={},kwargs={})'.format(self.name,self.args,self.kwargs)
# ----------------------------------------
# Command line parsing
# ----------------------------------------
TASK_RE=re.compile(r'(?P<name>[^()]+)(\((?P<args>.*)\))?')
def parseArguments(args):
i=0
parsed=[]
while i<len(args):
arg=args[i]
if arg.startswith('-'):
# although we say that these options are check that incompatible options aren't used together
if arg=='-V':
parsed.append(Version())
elif arg=='-h':
parsed.append(Help())
elif arg=='-b':
parsed.append(CreateBatchFile())
elif arg=='-l':
parsed.append(ListTasks())
elif arg=='-L':
parsed.append(ListTasks(includeDescription=False))
elif arg=='-m':
parsed.append(ModuleName(args[i+1]))
i+=1
elif arg=='-v':
parsed.append(Verbose())
elif arg=='-n':
parsed.append(DryRun())
elif arg=='-R':
parsed.append(CreatePieVenv())
parsed.append(UpdatePieVenv())
elif arg=='-r':
parsed.append(UpdatePieVenv())
elif arg=='-o':
if '=' not in args[i+1]: raise Exception('Option ("{}") must be in format name=value'.format(args[i+1]))
name,value=args[i+1].split('=')
parsed.append(Option(name,value))
i+=1
else:
raise Exception('Unknown argument: {}'.format(arg))
i+=1
else:
mo=TASK_RE.match(arg)
if mo:
taskArgs=mo.group('args')
taskArgs=taskArgs.split(',') if taskArgs else []
taskKwArgs={}
# and consume following -- args
i+=1
while i<len(args):
arg=args[i]
# kwargs must start with --
if not arg.startswith('--'): break
arg=arg[2:]
if '=' in arg:
name,value=arg.split('=')
else:
name=arg
value=True
name=name.replace('-','_')
taskKwArgs[name]=value
i+=1
# TODO: add further parsing to handle keyword arguments
parsed.append(TaskCall(mo.group('name'),args=taskArgs,kwargs=taskKwArgs))
else:
raise Exception('Unknown task format: {}'.format(arg))
return parsed
# ----------------------------------------
# pie venv
# ----------------------------------------
class PieVenv(venv):
PIE_REQUIREMENTS='requirements.pie.txt'
PIE_VENV='.venv-pie'
def __init__(self):
super(PieVenv,self).__init__(self.PIE_VENV)
def requirements_exists(self):
return os.path.isfile(self.PIE_REQUIREMENTS)
def create(self):
super(PieVenv,self).create('--system-site-packages')
def update(self):
self.pip_update()
self.pip_install_requirements(self.PIE_REQUIREMENTS)
def run_pie(self,args):
with venv(self.PIE_VENV):
r=cmd(r'python pie.py {}'.format(' '.join(args)))
return r
# ----------------------------------------
# entry point
# ----------------------------------------
def main(args):
parsed_args=parseArguments(args)
if parsed_args:
# only import tasks if needed, only run in a pie venv if needed, this saves exceptions and time when only looking for help or creating the batch file
if any([a.needsTasksImported for a in parsed_args]):
# run in the pie venv if required
pv=PieVenv()
# either no pie requirements or the venv is activated
if not pv.requirements_exists() or pv.is_activated():
# import tasks
try:
importTasks()
except Exception as e:
# pick up the specific case of not being able to find a pie_tasks module/package
# handle different attributes and messages in py2 vs py3
if isinstance(e,ImportError) and (getattr(e,'message','')=='No module named pie_tasks' or getattr(e,'msg','')=="No module named 'pie_tasks'"):
print('pie_tasks could not be found.',file=sys.stderr)
else:
print('An error occurred when importing pie_tasks:\n'+traceback.format_exc(),file=sys.stderr)
return 1
else:
# otherwise, we need to activate and run pie again, within the venv
if pv.exists():
return pv.run_pie(args)
else:
print('{} not found. You can create it with the -R argument.'.format(pv.PIE_VENV),file=sys.stderr)
return 1
# try to execute each arg
for a in parsed_args:
try:
a.execute()
except CmdContextManager.CmdError as e:
print('Error when executing command "{}". Errorcode = {}'.format(e.cmd,e.errorcode),file=sys.stderr)
return e.errorcode
except TaskCall.TaskNotFound as e:
print('Task {} could not be found.'.format(e.name),file=sys.stderr)
return 1
return 0
else:
Help().execute()
return 1
if __name__=='__main__':
# import pie so that both we and any pie_tasks code that imports pie are referring to the same module variables
import pie
# skip the name of the command
sys.exit(pie.main(sys.argv[1:]))