cli2 3.3.41__tar.gz → 3.3.44__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (45) hide show
  1. {cli2-3.3.41/cli2.egg-info → cli2-3.3.44}/PKG-INFO +1 -1
  2. {cli2-3.3.41 → cli2-3.3.44}/cli2/command.py +13 -14
  3. {cli2-3.3.41 → cli2-3.3.44}/cli2/group.py +23 -13
  4. {cli2-3.3.41 → cli2-3.3.44}/cli2/logging.py +4 -0
  5. {cli2-3.3.41 → cli2-3.3.44}/cli2/test_command.py +27 -0
  6. {cli2-3.3.41 → cli2-3.3.44}/cli2/test_group.py +8 -0
  7. {cli2-3.3.41 → cli2-3.3.44/cli2.egg-info}/PKG-INFO +1 -1
  8. {cli2-3.3.41 → cli2-3.3.44}/setup.py +1 -1
  9. {cli2-3.3.41 → cli2-3.3.44}/MANIFEST.in +0 -0
  10. {cli2-3.3.41 → cli2-3.3.44}/README.rst +0 -0
  11. {cli2-3.3.41 → cli2-3.3.44}/classifiers.txt +0 -0
  12. {cli2-3.3.41 → cli2-3.3.44}/cli2/__init__.py +0 -0
  13. {cli2-3.3.41 → cli2-3.3.44}/cli2/argument.py +0 -0
  14. {cli2-3.3.41 → cli2-3.3.44}/cli2/asyncio.py +0 -0
  15. {cli2-3.3.41 → cli2-3.3.44}/cli2/cli.py +0 -0
  16. {cli2-3.3.41 → cli2-3.3.44}/cli2/client.py +0 -0
  17. {cli2-3.3.41 → cli2-3.3.44}/cli2/colors.py +0 -0
  18. {cli2-3.3.41 → cli2-3.3.44}/cli2/configuration.py +0 -0
  19. {cli2-3.3.41 → cli2-3.3.44}/cli2/decorators.py +0 -0
  20. {cli2-3.3.41 → cli2-3.3.44}/cli2/display.py +0 -0
  21. {cli2-3.3.41 → cli2-3.3.44}/cli2/entry_point.py +0 -0
  22. {cli2-3.3.41 → cli2-3.3.44}/cli2/example_client.py +0 -0
  23. {cli2-3.3.41 → cli2-3.3.44}/cli2/example_client_complex.py +0 -0
  24. {cli2-3.3.41 → cli2-3.3.44}/cli2/example_nesting.py +0 -0
  25. {cli2-3.3.41 → cli2-3.3.44}/cli2/example_obj.py +0 -0
  26. {cli2-3.3.41 → cli2-3.3.44}/cli2/node.py +0 -0
  27. {cli2-3.3.41 → cli2-3.3.44}/cli2/overrides.py +0 -0
  28. {cli2-3.3.41 → cli2-3.3.44}/cli2/sphinx.py +0 -0
  29. {cli2-3.3.41 → cli2-3.3.44}/cli2/table.py +0 -0
  30. {cli2-3.3.41 → cli2-3.3.44}/cli2/test.py +0 -0
  31. {cli2-3.3.41 → cli2-3.3.44}/cli2/test_cli.py +0 -0
  32. {cli2-3.3.41 → cli2-3.3.44}/cli2/test_client.py +0 -0
  33. {cli2-3.3.41 → cli2-3.3.44}/cli2/test_configuration.py +0 -0
  34. {cli2-3.3.41 → cli2-3.3.44}/cli2/test_decorators.py +0 -0
  35. {cli2-3.3.41 → cli2-3.3.44}/cli2/test_display.py +0 -0
  36. {cli2-3.3.41 → cli2-3.3.44}/cli2/test_entry_point.py +0 -0
  37. {cli2-3.3.41 → cli2-3.3.44}/cli2/test_inject.py +0 -0
  38. {cli2-3.3.41 → cli2-3.3.44}/cli2/test_node.py +0 -0
  39. {cli2-3.3.41 → cli2-3.3.44}/cli2/test_table.py +0 -0
  40. {cli2-3.3.41 → cli2-3.3.44}/cli2.egg-info/SOURCES.txt +0 -0
  41. {cli2-3.3.41 → cli2-3.3.44}/cli2.egg-info/dependency_links.txt +0 -0
  42. {cli2-3.3.41 → cli2-3.3.44}/cli2.egg-info/entry_points.txt +0 -0
  43. {cli2-3.3.41 → cli2-3.3.44}/cli2.egg-info/requires.txt +0 -0
  44. {cli2-3.3.41 → cli2-3.3.44}/cli2.egg-info/top_level.txt +0 -0
  45. {cli2-3.3.41 → cli2-3.3.44}/setup.cfg +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.2
2
2
  Name: cli2
3
- Version: 3.3.41
3
+ Version: 3.3.44
4
4
  Summary: image:: https://yourlabs.io/oss/cli2/badges/master/pipeline.svg
5
5
  Home-page: https://yourlabs.io/oss/cli2
6
6
  Author: James Pic
@@ -1,6 +1,5 @@
1
1
  import asyncio
2
2
  import inspect
3
- import sys
4
3
 
5
4
  from docstring_parser import parse
6
5
 
@@ -278,7 +277,14 @@ class Command(EntryPoint, dict):
278
277
  return self.help()
279
278
 
280
279
  if self.async_mode():
281
- return asyncio.run(self.async_call(*argv))
280
+ try:
281
+ return asyncio.run(self.async_call(*argv))
282
+ except KeyboardInterrupt:
283
+ print('exiting cleanly...')
284
+ self.exit_code = 1
285
+ return
286
+ finally:
287
+ self.post_result = asyncio.run(async_resolve(self.post_call()))
282
288
 
283
289
  error = self.parse(*argv)
284
290
  if error:
@@ -296,12 +302,12 @@ class Command(EntryPoint, dict):
296
302
  for _ in result:
297
303
  display.print(_)
298
304
  result = None
305
+ return result
299
306
  except KeyboardInterrupt:
300
- print('exiting')
301
- sys.exit(1)
307
+ print('exiting cleanly...')
308
+ self.exit_code = 1
302
309
  finally:
303
310
  self.post_result = self.post_call()
304
- return result
305
311
 
306
312
  async def async_call(self, *argv):
307
313
  """ Call with async stuff in single event loop """
@@ -324,15 +330,8 @@ class Command(EntryPoint, dict):
324
330
  for _, arg in enumerate(factories):
325
331
  arg.value = results[_]
326
332
 
327
- try:
328
- result = self.call(*self.bound.args, **self.bound.kwargs)
329
- result = await async_resolve(result, output=True)
330
- except KeyboardInterrupt:
331
- print('exiting')
332
- sys.exit(1)
333
- finally:
334
- self.post_result = await async_resolve(self.post_call())
335
- return result
333
+ result = self.call(*self.bound.args, **self.bound.kwargs)
334
+ return await async_resolve(result, output=True)
336
335
 
337
336
  def ordered(self, factories=False):
338
337
  """
@@ -195,19 +195,29 @@ class Group(EntryPoint, dict):
195
195
  for name, method in cls.__dict__.items():
196
196
  if leaf and getattr(final, name, '_') is None:
197
197
  continue
198
- wrapped_method = getattr(method, '__func__', None)
199
- cfg = getattr(
200
- wrapped_method,
201
- 'cli2',
202
- getattr(method, 'cli2', None),
203
- )
204
- if cfg is None:
205
- continue
206
- condition = cfg.get('condition', None)
207
- if condition:
208
- if not condition(final):
209
- continue
210
- self.cmd(method)
198
+ self.load_method(final, method)
199
+
200
+ def load_obj(self, obj):
201
+ """
202
+ Load all methods which have been decorated with @cmd
203
+ """
204
+ for name in dir(obj):
205
+ self.load_method(obj, getattr(obj, name))
206
+
207
+ def load_method(self, obj, method):
208
+ wrapped_method = getattr(method, '__func__', None)
209
+ cfg = getattr(
210
+ wrapped_method,
211
+ 'cli2',
212
+ getattr(method, 'cli2', None),
213
+ )
214
+ if cfg is None:
215
+ return
216
+ condition = cfg.get('condition', None)
217
+ if condition:
218
+ if not condition(obj):
219
+ return
220
+ self.cmd(method)
211
221
 
212
222
  def __call__(self, *argv):
213
223
  self.exit_code = 0
@@ -22,6 +22,10 @@ class YAMLFormatter:
22
22
 
23
23
  def configure():
24
24
  LOG_LEVEL = os.getenv('LOG_LEVEL', 'WARNING').upper()
25
+
26
+ if os.getenv('DEBUG'):
27
+ LOG_LEVEL = 'DEBUG'
28
+
25
29
  timestamper = structlog.processors.TimeStamper(fmt='%Y-%m-%d %H:%M:%S')
26
30
  pre_chain = [
27
31
  # add log level and timestamp to event_dict
@@ -1,3 +1,4 @@
1
+ import asyncio
1
2
  import inspect
2
3
  import pytest
3
4
  import os
@@ -684,3 +685,29 @@ def test_cli2():
684
685
 
685
686
  cmd = Command(test)
686
687
  assert cmd('x') == ('x', cmd)
688
+
689
+
690
+ def test_keyboard_interrupt():
691
+ def func():
692
+ raise KeyboardInterrupt()
693
+
694
+ cmd = Command(func)
695
+ cmd()
696
+ assert cmd.exit_code == 1
697
+
698
+
699
+ def test_keyboard_interrupt_async():
700
+ class YourCommand(Command):
701
+ async def post_call(self):
702
+ return 'foo'
703
+
704
+ async def foo():
705
+ raise KeyboardInterrupt()
706
+
707
+ async def func():
708
+ await asyncio.gather(foo())
709
+
710
+ cmd = YourCommand(func)
711
+ cmd()
712
+ assert cmd.exit_code == 1
713
+ assert cmd.post_result == 'foo'
@@ -232,9 +232,17 @@ def test_load_cls():
232
232
  group.load_cls(Foo)
233
233
  assert list(group.keys()) == ['help', 'test', 'test2']
234
234
 
235
+ group = Group()
236
+ group.load_obj(Foo())
237
+ assert list(group.keys()) == ['help', 'test', 'test2']
238
+
235
239
  class Child(Foo):
236
240
  test2 = None
237
241
 
238
242
  group = Group()
239
243
  group.load_cls(Child)
240
244
  assert list(group.keys()) == ['help', 'test']
245
+
246
+ group = Group()
247
+ group.load_obj(Child())
248
+ assert list(group.keys()) == ['help', 'test']
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.2
2
2
  Name: cli2
3
- Version: 3.3.41
3
+ Version: 3.3.44
4
4
  Summary: image:: https://yourlabs.io/oss/cli2/badges/master/pipeline.svg
5
5
  Home-page: https://yourlabs.io/oss/cli2
6
6
  Author: James Pic
@@ -3,7 +3,7 @@ from setuptools import setup
3
3
 
4
4
  setup(
5
5
  name='cli2',
6
- version='3.3.41',
6
+ version='3.3.44',
7
7
  setup_requires='setupmeta',
8
8
  install_requires=[
9
9
  'docstring_parser',
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes