basic runner format; write a MAIN function
introduce --verbosity flag rather than mixed logging settings; correct color misnaming to ANSI convention; added sanity-check; simplified hello-world; created FZF_USER_INPUT to replace the confusing FZF_HEAD and FZF_TAIL swap INFO for DEBUG v3-to-v4 upgrade docs bring some much-needed tender love and care to the scwrypts runner improved i/o handling on the run executable means this is no longer relevant FINALLY fix the weird cases for zsh/read builtin (particularly around reading one character from tty/pipe/file); also gave a --force-user-input flag in case you want to require user input on a yn prompt update ZLE plugin so it no more make errors FZF_(HEAD|TAIL) refactor to FZF_USER_INPUT plugins/kubectl migration from v3 to v4 plugins/ci migration from v3 to v4 refactor py/lib into python-scwrypts subproject verbosity is stupid lets call it log-level fix bug with virtualenv loading mergedeep to slow so I made my options dict shallow hokay first iteration of python-dudes is ready circleci configuration for python builds npm package for scwrypts 3.9.1 initial build/test steps for nodejs go go ok ok fix output ok ok finalize publish steps
This commit is contained in:
		
							
								
								
									
										1
									
								
								py/lib/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								py/lib/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| dist/ | ||||
							
								
								
									
										3
									
								
								py/lib/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								py/lib/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| # Python Scwrypts | ||||
| [](https://python.org) | ||||
| <br> | ||||
| @@ -1,6 +0,0 @@ | ||||
| import py.lib.data | ||||
| import py.lib.fzf | ||||
| import py.lib.http | ||||
| import py.lib.redis | ||||
| import py.lib.scwrypts | ||||
| import py.lib.twilio | ||||
| @@ -1 +0,0 @@ | ||||
| import py.lib.data.converter | ||||
| @@ -1 +0,0 @@ | ||||
| from py.lib.fzf.client import fzf, fzf_tail, fzf_head | ||||
| @@ -1,5 +0,0 @@ | ||||
| from py.lib.http.client import get_request_client | ||||
|  | ||||
| import py.lib.http.directus | ||||
| import py.lib.http.discord | ||||
| import py.lib.http.linear | ||||
| @@ -1,20 +0,0 @@ | ||||
| from requests import request | ||||
|  | ||||
|  | ||||
| def get_request_client(base_url, headers=None): | ||||
|     if headers is None: | ||||
|         headers = {} | ||||
|  | ||||
|     return lambda method, endpoint, **kwargs: request( | ||||
|             method = method, | ||||
|             url = f'{base_url}/{endpoint}', | ||||
|             headers = { | ||||
|                 **headers, | ||||
|                 **kwargs.get('headers', {}), | ||||
|                 }, | ||||
|             **{ | ||||
|                 key: value | ||||
|                 for key, value in kwargs.items() | ||||
|                 if key != 'headers' | ||||
|                 }, | ||||
|             ) | ||||
| @@ -1,2 +0,0 @@ | ||||
| from py.lib.http.directus.client import * | ||||
| from py.lib.http.directus.constant import * | ||||
| @@ -1,56 +0,0 @@ | ||||
| from py.lib.http import get_request_client | ||||
| from py.lib.scwrypts import getenv | ||||
|  | ||||
|  | ||||
| REQUEST     = None | ||||
| COLLECTIONS = None | ||||
| FIELDS      = {} | ||||
|  | ||||
|  | ||||
| def request(method, endpoint, **kwargs): | ||||
|     global REQUEST # pylint: disable=global-statement | ||||
|  | ||||
|     if REQUEST is None: | ||||
|         REQUEST = get_request_client( | ||||
|                 base_url = getenv("DIRECTUS__BASE_URL"), | ||||
|                 headers = { | ||||
|                     'Authorization': f'bearer {getenv("DIRECTUS__API_TOKEN")}', | ||||
|                     } | ||||
|                 ) | ||||
|  | ||||
|     return REQUEST(method, endpoint, **kwargs) | ||||
|  | ||||
| def graphql(query, system=False): | ||||
|     return request( | ||||
|             'POST', | ||||
|             'graphql' if system is True else 'graphql/system', | ||||
|             json={'query': query}, | ||||
|             ) | ||||
|  | ||||
|  | ||||
| def get_collections(): | ||||
|     global COLLECTIONS # pylint: disable=global-statement | ||||
|  | ||||
|     if COLLECTIONS is None: | ||||
|         COLLECTIONS = [ | ||||
|             item['collection'] | ||||
|             for item in request( | ||||
|                 'GET', | ||||
|                 'collections?limit=-1&fields[]=collection', | ||||
|                 ).json()['data'] | ||||
|             ] | ||||
|  | ||||
|     return COLLECTIONS | ||||
|  | ||||
|  | ||||
| def get_fields(collection): | ||||
|     if FIELDS.get(collection) is None: | ||||
|         FIELDS[collection] = [ | ||||
|                 item['field'] | ||||
|                 for item in request( | ||||
|                     'GET', | ||||
|                     f'fields/{collection}?limit=-1&fields[]=field', | ||||
|                     ).json()['data'] | ||||
|                 ] | ||||
|  | ||||
|     return FIELDS[collection] | ||||
| @@ -1,2 +0,0 @@ | ||||
| from py.lib.http.discord.client import * | ||||
| from py.lib.http.discord.send_message import * | ||||
| @@ -1,20 +0,0 @@ | ||||
| from py.lib.http import get_request_client | ||||
| from py.lib.scwrypts import getenv | ||||
|  | ||||
| REQUEST = None | ||||
|  | ||||
| def request(method, endpoint, **kwargs): | ||||
|     global REQUEST # pylint: disable=global-statement | ||||
|  | ||||
|     if REQUEST is None: | ||||
|         headers = {} | ||||
|  | ||||
|         if (token := getenv("DISCORD__BOT_TOKEN", required = False)) is not None: | ||||
|             headers['Authorization'] = f'Bot {token}' | ||||
|  | ||||
|         REQUEST = get_request_client( | ||||
|                 base_url = 'https://discord.com/api', | ||||
|                 headers = headers, | ||||
|                 ) | ||||
|  | ||||
|     return REQUEST(method, endpoint, **kwargs) | ||||
| @@ -1 +0,0 @@ | ||||
| from py.lib.http.linear.client import * | ||||
| @@ -1,20 +0,0 @@ | ||||
| from py.lib.http import get_request_client | ||||
| from py.lib.scwrypts import getenv | ||||
|  | ||||
| REQUEST = None | ||||
|  | ||||
| def request(method, endpoint, **kwargs): | ||||
|     global REQUEST # pylint: disable=global-statement | ||||
|  | ||||
|     if REQUEST is None: | ||||
|         REQUEST = get_request_client( | ||||
|                 base_url = 'https://api.linear.app', | ||||
|                 headers = { | ||||
|                     'Authorization': f'bearer {getenv("LINEAR__API_TOKEN")}', | ||||
|                     } | ||||
|                 ) | ||||
|  | ||||
|     return REQUEST(method, endpoint, **kwargs) | ||||
|  | ||||
| def graphql(query): | ||||
|     return request('POST', 'graphql', json={'query': query}) | ||||
							
								
								
									
										60
									
								
								py/lib/pyproject.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								py/lib/pyproject.toml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| [project] | ||||
| name = 'scwrypts' | ||||
| description = 'scwrypts library and invoker' | ||||
| license = 'GPL-3.0-or-later' | ||||
|  | ||||
| readme = 'README.md' | ||||
| requires-python = '>=3.10' | ||||
|  | ||||
| authors = [ | ||||
| 	{ name='yage', email='yage@yage.io' }, | ||||
| ] | ||||
|  | ||||
|  | ||||
| classifiers = [ | ||||
| 	'Programming Language :: Python :: 3', | ||||
| 	'Programming Language :: Python :: 3.10', | ||||
| 	'Programming Language :: Python :: 3.11', | ||||
| 	'Programming Language :: Python :: 3.12', | ||||
| 	'License :: OSI Approved :: GNU General Public License v3 or later (GPLv3+)', | ||||
| 	] | ||||
|  | ||||
| dynamic = ['version'] | ||||
|  | ||||
| dependencies = [ | ||||
| 	'bpython', | ||||
| 	'pyfzf', | ||||
| 	'pyyaml', | ||||
| 	'redis', | ||||
| 	'twilio', | ||||
| ] | ||||
|  | ||||
| [project.optional-dependencies] | ||||
| dev = [ | ||||
| 	'pylint', | ||||
| ] | ||||
|  | ||||
| test = [ | ||||
| 	'pytest', | ||||
| ] | ||||
|  | ||||
| [project.urls] | ||||
| homepage = 'https://github.com/wrynegade/scwrypts' | ||||
| issues = 'https://github.com/wrynegade/scwrypts/issues' | ||||
|  | ||||
| [build-system] | ||||
| requires = [ | ||||
| 	'hatchling', | ||||
| 	'versioningit', | ||||
| ] | ||||
| build-backend = 'hatchling.build' | ||||
|  | ||||
| [tool.hatch.version] | ||||
| source = 'versioningit' | ||||
|  | ||||
| [tool.hatch.build.targets.wheel] | ||||
| packages = ['./'] | ||||
|  | ||||
| [tool.versioningit] | ||||
| match = ['v*'] | ||||
|  | ||||
| @@ -1 +0,0 @@ | ||||
| from py.lib.redis.client import get_client | ||||
| @@ -1,6 +1,13 @@ | ||||
| from py.lib.scwrypts.execute import execute | ||||
| from py.lib.scwrypts.getenv import getenv | ||||
| from py.lib.scwrypts.interactive import interactive | ||||
| from py.lib.scwrypts.run import run | ||||
| ''' | ||||
| scwrypts | ||||
|  | ||||
| import py.lib.scwrypts.io | ||||
| python library functions and invoker for scwrypts | ||||
| ''' | ||||
|  | ||||
| __all__ = [ | ||||
|         'scwrypts', | ||||
|         'execute', | ||||
|         'interactive', | ||||
|         ] | ||||
|  | ||||
| from .scwrypts import scwrypts, execute, interactive | ||||
|   | ||||
							
								
								
									
										1
									
								
								py/lib/scwrypts/data/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								py/lib/scwrypts/data/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| from .converter import convert | ||||
| @@ -4,9 +4,6 @@ import yaml | ||||
| 
 | ||||
| 
 | ||||
| def convert(input_stream, input_type, output_stream, output_type): | ||||
|     if input_type == output_type: | ||||
|         raise ValueError('input type and output type are the same') | ||||
| 
 | ||||
|     data = convert_input(input_stream, input_type) | ||||
|     write_output(output_stream, output_type, data) | ||||
| 
 | ||||
							
								
								
									
										73
									
								
								py/lib/scwrypts/data/test_converter.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								py/lib/scwrypts/data/test_converter.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,73 @@ | ||||
| from io import StringIO | ||||
|  | ||||
| from pytest import raises | ||||
|  | ||||
| from scwrypts.test import generate | ||||
|  | ||||
| from .converter import convert | ||||
|  | ||||
| GENERATE_OPTIONS = { | ||||
|         'depth': 1, | ||||
|         'minimum': -999999, | ||||
|         'maximum':  999999, | ||||
|         'dict_key_types': {str, int}, | ||||
|         'csv_columns_minimum': 10, | ||||
|         'csv_columns_maximum': 64, | ||||
|         'csv_rows_minimum': 10, | ||||
|         'csv_rows_maximum': 64, | ||||
|         } | ||||
|  | ||||
| INPUT_TYPES  = {'csv', 'json', 'yaml'} | ||||
| OUTPUT_TYPES = {'csv', 'json', 'yaml'} | ||||
|  | ||||
|  | ||||
| def test_convert_to_csv(): | ||||
|     for input_type in INPUT_TYPES: | ||||
|         input_stream = generate(input_type, { | ||||
|             **GENERATE_OPTIONS, | ||||
|             'data_types': {bool,int,float,str}, | ||||
|             }) | ||||
|  | ||||
|         if isinstance(input_stream, str): | ||||
|             input_stream = StringIO(input_stream) | ||||
|  | ||||
|  | ||||
|         convert(input_stream, input_type, StringIO(), 'csv') | ||||
|  | ||||
| def test_convert_to_json(): | ||||
|     for input_type in INPUT_TYPES: | ||||
|         input_stream = generate(input_type, GENERATE_OPTIONS) | ||||
|  | ||||
|         if isinstance(input_stream, str): | ||||
|             input_stream = StringIO(input_stream) | ||||
|  | ||||
|         convert(input_stream, input_type, StringIO(), 'json') | ||||
|  | ||||
| def test_convert_to_yaml(): | ||||
|     for input_type in INPUT_TYPES: | ||||
|         input_stream = generate(input_type, GENERATE_OPTIONS) | ||||
|  | ||||
|         if isinstance(input_stream, str): | ||||
|             input_stream = StringIO(input_stream) | ||||
|  | ||||
|         convert(input_stream, input_type, StringIO(), 'yaml') | ||||
|  | ||||
|  | ||||
| def test_convert_deep_json_to_yaml(): | ||||
|     input_stream = generate('json', {**GENERATE_OPTIONS, 'depth': 4}) | ||||
|     convert(input_stream, 'json', StringIO(), 'yaml') | ||||
|  | ||||
| def test_convert_deep_yaml_to_json(): | ||||
|     input_stream = generate('yaml', {**GENERATE_OPTIONS, 'depth': 4}) | ||||
|     convert(input_stream, 'yaml', StringIO(), 'json') | ||||
|  | ||||
|  | ||||
| def test_convert_output_unsupported(): | ||||
|     for input_type in list(INPUT_TYPES): | ||||
|         with raises(ValueError): | ||||
|             convert(StringIO(), input_type, StringIO(), generate(str)) | ||||
|  | ||||
| def test_convert_input_unsupported(): | ||||
|     for output_type in list(OUTPUT_TYPES): | ||||
|         with raises(ValueError): | ||||
|             convert(StringIO(), generate(str), StringIO(), output_type) | ||||
| @@ -1,6 +1,6 @@ | ||||
| from os import getenv as os_getenv | ||||
| 
 | ||||
| from py.lib.scwrypts.exceptions import MissingVariableError | ||||
| from .scwrypts.exceptions import MissingVariableError | ||||
| 
 | ||||
| 
 | ||||
| def getenv(name, required=True): | ||||
| @@ -1,16 +0,0 @@ | ||||
| from argparse import ArgumentError | ||||
|  | ||||
|  | ||||
| class MissingVariableError(EnvironmentError): | ||||
|     def init(self, name): | ||||
|         super().__init__(f'Missing required environment variable "{name}"') | ||||
|  | ||||
|  | ||||
| class ImportedExecutableError(ImportError): | ||||
|     def __init__(self): | ||||
|         super().__init__('executable only; must run through scwrypts') | ||||
|  | ||||
|  | ||||
| class MissingFlagAndEnvironmentVariableError(EnvironmentError, ArgumentError): | ||||
|     def __init__(self, flags, env_var): | ||||
|         super().__init__(f'must provide at least one of : {{ flags: {flags} OR {env_var} }}') | ||||
							
								
								
									
										1
									
								
								py/lib/scwrypts/fzf/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								py/lib/scwrypts/fzf/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| from .client import fzf, fzf_tail, fzf_head | ||||
							
								
								
									
										1
									
								
								py/lib/scwrypts/http/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								py/lib/scwrypts/http/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| from .client import get_request_client | ||||
							
								
								
									
										25
									
								
								py/lib/scwrypts/http/client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								py/lib/scwrypts/http/client.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| from requests import request | ||||
|  | ||||
|  | ||||
| CLIENTS = {} | ||||
|  | ||||
| def get_request_client(base_url, headers=None): | ||||
|     if CLIENTS.get(base_url, None) is None: | ||||
|         if headers is None: | ||||
|             headers = {} | ||||
|  | ||||
|         CLIENTS[base_url] = lambda method, endpoint, **kwargs: request( | ||||
|                 method = method, | ||||
|                 url = f'{base_url}/{endpoint}', | ||||
|                 headers = { | ||||
|                     **headers, | ||||
|                     **kwargs.get('headers', {}), | ||||
|                     }, | ||||
|                 **{ | ||||
|                     key: value | ||||
|                     for key, value in kwargs.items() | ||||
|                     if key != 'headers' | ||||
|                     }, | ||||
|                 ) | ||||
|  | ||||
|     return CLIENTS[base_url] | ||||
							
								
								
									
										43
									
								
								py/lib/scwrypts/http/conftest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								py/lib/scwrypts/http/conftest.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| from types import SimpleNamespace | ||||
|  | ||||
| from pytest import fixture | ||||
|  | ||||
| from scwrypts.test import generate | ||||
| from scwrypts.test.character_set import uri | ||||
|  | ||||
| options = { | ||||
|         'str_length_minimum':   8, | ||||
|         'str_length_maximum': 128, | ||||
|         'uuid_output_type':   str, | ||||
|         } | ||||
|  | ||||
| def get_request_client_sample_data(): | ||||
|     return { | ||||
|             'base_url' : generate(str, options | {'character_set': uri}), | ||||
|             'endpoint' : generate(str, options | {'character_set': uri}), | ||||
|             'method'   : generate(str, options), | ||||
|             'response' : generate('requests_Response', options | {'depth': 4}), | ||||
|             'payload'  : generate(dict, { | ||||
|                 **options, | ||||
|                 'depth': 1, | ||||
|                 'data_types': { str, 'uuid' }, | ||||
|                 }), | ||||
|             } | ||||
|  | ||||
| @fixture(name='sample') | ||||
| def fixture_sample(): | ||||
|     return SimpleNamespace( | ||||
|             **get_request_client_sample_data(), | ||||
|  | ||||
|             headers = generate(dict, { | ||||
|                 **options, | ||||
|                 'depth': 1, | ||||
|                 'data_types': { str, 'uuid' }, | ||||
|                 }), | ||||
|  | ||||
|             payload_headers = generate(dict, { | ||||
|                 **options, | ||||
|                 'depth': 1, | ||||
|                 'data_types': { str, 'uuid' }, | ||||
|                 }), | ||||
|             ) | ||||
| @@ -1,3 +1,23 @@ | ||||
| ''' | ||||
| basic scwrypts.http client for directus | ||||
| 
 | ||||
| configured by setting DIRECTUS__BASE_URL and DIRECTUS__API_TOKEN in | ||||
| scwrypts environment | ||||
| ''' | ||||
| 
 | ||||
| __all__ = [ | ||||
|         'request', | ||||
|         'graphql', | ||||
|         'get_collections', | ||||
|         'get_fields', | ||||
|         'FILTER_OPERATORS', | ||||
|         ] | ||||
| 
 | ||||
| from .client import request | ||||
| from .graphql import graphql | ||||
| from .collections import get_collections | ||||
| from .fields import get_fields | ||||
| 
 | ||||
| FILTER_OPERATORS = { | ||||
|         '_eq', | ||||
|         '_neq', | ||||
							
								
								
									
										12
									
								
								py/lib/scwrypts/http/directus/client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								py/lib/scwrypts/http/directus/client.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| from scwrypts.env import getenv | ||||
|  | ||||
| from .. import get_request_client | ||||
|  | ||||
|  | ||||
| def request(method, endpoint, **kwargs): | ||||
|     return get_request_client( | ||||
|             base_url = getenv("DIRECTUS__BASE_URL"), | ||||
|             headers = { | ||||
|                 'Authorization': f'bearer {getenv("DIRECTUS__API_TOKEN")}', | ||||
|                 } | ||||
|             )(method, endpoint, **kwargs) | ||||
							
								
								
									
										18
									
								
								py/lib/scwrypts/http/directus/collections.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								py/lib/scwrypts/http/directus/collections.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| from .client import request | ||||
|  | ||||
|  | ||||
| COLLECTIONS = None | ||||
|  | ||||
| def get_collections(): | ||||
|     global COLLECTIONS # pylint: disable=global-statement | ||||
|  | ||||
|     if COLLECTIONS is None: | ||||
|         COLLECTIONS = [ | ||||
|             item['collection'] | ||||
|             for item in request( | ||||
|                 'GET', | ||||
|                 'collections?limit=-1&fields[]=collection', | ||||
|                 ).json()['data'] | ||||
|             ] | ||||
|  | ||||
|     return COLLECTIONS | ||||
							
								
								
									
										16
									
								
								py/lib/scwrypts/http/directus/conftest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								py/lib/scwrypts/http/directus/conftest.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| from types import SimpleNamespace | ||||
|  | ||||
| from pytest import fixture | ||||
|  | ||||
| from scwrypts.test import generate | ||||
| from scwrypts.test.character_set import uri | ||||
| from ..conftest import options, get_request_client_sample_data | ||||
|  | ||||
|  | ||||
| @fixture(name='sample') | ||||
| def fixture_sample(): | ||||
|     return SimpleNamespace( | ||||
|             **get_request_client_sample_data(), | ||||
|             api_token = generate(str, options | {'character_set': uri}), | ||||
|             query     = generate(str, options), | ||||
|             ) | ||||
							
								
								
									
										16
									
								
								py/lib/scwrypts/http/directus/fields.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								py/lib/scwrypts/http/directus/fields.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| from .client import request | ||||
|  | ||||
|  | ||||
| FIELDS      = {} | ||||
|  | ||||
| def get_fields(collection): | ||||
|     if FIELDS.get(collection) is None: | ||||
|         FIELDS[collection] = [ | ||||
|                 item['field'] | ||||
|                 for item in request( | ||||
|                     'GET', | ||||
|                     f'fields/{collection}?limit=-1&fields[]=field', | ||||
|                     ).json()['data'] | ||||
|                 ] | ||||
|  | ||||
|     return FIELDS[collection] | ||||
							
								
								
									
										9
									
								
								py/lib/scwrypts/http/directus/graphql.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								py/lib/scwrypts/http/directus/graphql.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| from .client import request | ||||
|  | ||||
|  | ||||
| def graphql(query, system=False): | ||||
|     return request( | ||||
|             'POST', | ||||
|             'graphql' if system is False else 'graphql/system', | ||||
|             json={'query': query}, | ||||
|             ) | ||||
							
								
								
									
										43
									
								
								py/lib/scwrypts/http/directus/test_client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								py/lib/scwrypts/http/directus/test_client.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| from unittest.mock import patch | ||||
|  | ||||
| from pytest import fixture | ||||
|  | ||||
| from .client import request | ||||
|  | ||||
|  | ||||
| def test_directus_request(sample, _response): | ||||
|     assert _response == sample.response | ||||
|  | ||||
| def test_directus_request_client_setup(sample, _response, mock_get_request_client): | ||||
|     mock_get_request_client.assert_called_once_with( | ||||
|             base_url = sample.base_url, | ||||
|             headers = { 'Authorization': f'bearer {sample.api_token}' }, | ||||
|             ) | ||||
|  | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| @fixture(name='_response') | ||||
| def fixture_response(sample): | ||||
|     return request( | ||||
|             method   = sample.method, | ||||
|             endpoint = sample.endpoint, | ||||
|             **sample.payload, | ||||
|             ) | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| @fixture(name='mock_getenv', autouse=True) | ||||
| def fixture_mock_getenv(sample): | ||||
|     with patch('scwrypts.http.directus.client.getenv',) as mock: | ||||
|         mock.side_effect = lambda name: { | ||||
|                 'DIRECTUS__BASE_URL': sample.base_url, | ||||
|                 'DIRECTUS__API_TOKEN': sample.api_token, | ||||
|                 }[name] | ||||
|         yield mock | ||||
|  | ||||
| @fixture(name='mock_get_request_client', autouse=True) | ||||
| def fixture_mock_get_request_client(sample): | ||||
|     with patch('scwrypts.http.directus.client.get_request_client') as mock: | ||||
|         mock.return_value = lambda method, endpoint, **kwargs: sample.response | ||||
|         yield mock | ||||
							
								
								
									
										45
									
								
								py/lib/scwrypts/http/directus/test_graphql.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								py/lib/scwrypts/http/directus/test_graphql.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| from unittest.mock import patch | ||||
|  | ||||
| from pytest import fixture | ||||
|  | ||||
| from .graphql import graphql | ||||
|  | ||||
|  | ||||
| def test_directus_graphql(sample, _response, _mock_request): | ||||
|     assert _response == sample.response | ||||
|  | ||||
| def test_directus_graphql_request_payload(sample, _response, _mock_request): | ||||
|     _mock_request.assert_called_once_with( | ||||
|             'POST', | ||||
|             'graphql', | ||||
|             json = {'query': sample.query}, | ||||
|             ) | ||||
|  | ||||
| def test_directus_graphql_system(sample, _response_system): | ||||
|     assert _response_system == sample.response | ||||
|  | ||||
| def test_directus_graphql_system_request_payload(sample, _response_system, _mock_request): | ||||
|     _mock_request.assert_called_once_with( | ||||
|             'POST', | ||||
|             'graphql/system', | ||||
|             json = {'query': sample.query}, | ||||
|             ) | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| @fixture(name='_response') | ||||
| def fixture_response(sample, _mock_request): | ||||
|     return graphql(sample.query) | ||||
|  | ||||
|  | ||||
| @fixture(name='_response_system') | ||||
| def fixture_response_system(sample, _mock_request): | ||||
|     return graphql(sample.query, system=True) | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| @fixture(name='_mock_request') | ||||
| def fixture_mock_request(sample): | ||||
|     with patch('scwrypts.http.directus.graphql.request') as mock: | ||||
|         mock.return_value = sample.response | ||||
|         yield mock | ||||
							
								
								
									
										14
									
								
								py/lib/scwrypts/http/discord/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								py/lib/scwrypts/http/discord/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| ''' | ||||
| basic scwrypts.http client for discord | ||||
|  | ||||
| configured by setting various DISCORD__* options in the | ||||
| scwrypts environment | ||||
| ''' | ||||
|  | ||||
| __all__ = [ | ||||
|         'request', | ||||
|         'send_message', | ||||
|         ] | ||||
|  | ||||
| from .client import request | ||||
| from .send_message import send_message | ||||
							
								
								
									
										15
									
								
								py/lib/scwrypts/http/discord/client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								py/lib/scwrypts/http/discord/client.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| from scwrypts.env import getenv | ||||
|  | ||||
| from .. import get_request_client | ||||
|  | ||||
|  | ||||
| def request(method, endpoint, **kwargs): | ||||
|     headers = {} | ||||
|  | ||||
|     if (token := getenv("DISCORD__BOT_TOKEN", required = False)) is not None: | ||||
|         headers['Authorization'] = f'Bot {token}' | ||||
|  | ||||
|     return get_request_client( | ||||
|             base_url = 'https://discord.com/api', | ||||
|             headers = headers, | ||||
|             )(method, endpoint, **kwargs) | ||||
							
								
								
									
										25
									
								
								py/lib/scwrypts/http/discord/conftest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								py/lib/scwrypts/http/discord/conftest.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| from string import ascii_letters, digits | ||||
| from types import SimpleNamespace | ||||
|  | ||||
| from pytest import fixture | ||||
|  | ||||
| from scwrypts.test import generate | ||||
| from scwrypts.test.character_set import uri | ||||
| from ..conftest import options, get_request_client_sample_data | ||||
|  | ||||
| @fixture(name='sample') | ||||
| def fixture_sample(): | ||||
|     return SimpleNamespace( | ||||
|             **{ | ||||
|                 **get_request_client_sample_data(), | ||||
|                 'base_url': 'https://discord.com/api', | ||||
|                 }, | ||||
|             bot_token  = generate(str, options | {'character_set': uri}), | ||||
|             username   = generate(str, options | {'character_set': ascii_letters + digits}), | ||||
|             avatar_url = generate(str, options | {'character_set': uri}), | ||||
|             webhook    = generate(str, options | {'character_set': uri}), | ||||
|             channel_id = generate(str, options | {'character_set': uri}), | ||||
|             content_header = generate(str, options), | ||||
|             content_footer = generate(str, options), | ||||
|             content = generate(str, options), | ||||
|         ) | ||||
| @@ -1,5 +1,5 @@ | ||||
| from py.lib.scwrypts import getenv | ||||
| from py.lib.http.discord import request | ||||
| from scwrypts.env import getenv | ||||
| from .client import request | ||||
| 
 | ||||
| def send_message(content, channel_id=None, webhook=None, username=None, avatar_url=None, **kwargs): | ||||
|     if username is None: | ||||
							
								
								
									
										54
									
								
								py/lib/scwrypts/http/discord/test_client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								py/lib/scwrypts/http/discord/test_client.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | ||||
| from unittest.mock import patch | ||||
|  | ||||
| from pytest import fixture | ||||
|  | ||||
| from .client import request | ||||
|  | ||||
|  | ||||
| def test_discord_request(sample, _response): | ||||
|     assert _response == sample.response | ||||
|  | ||||
| def test_discord_request_client_setup(sample, mock_get_request_client, _mock_getenv, _response): | ||||
|     mock_get_request_client.assert_called_once_with( | ||||
|             base_url = sample.base_url, | ||||
|             headers = { 'Authorization': f'Bot {sample.bot_token}' }, | ||||
|             ) | ||||
|  | ||||
| def test_discord_request_client_setup_public(sample, mock_get_request_client, _mock_getenv_optional, _response): | ||||
|     mock_get_request_client.assert_called_once_with( | ||||
|             base_url = sample.base_url, | ||||
|             headers = {}, | ||||
|             ) | ||||
|  | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| @fixture(name='_response') | ||||
| def fixture_response(sample): | ||||
|     return request( | ||||
|             method   = sample.method, | ||||
|             endpoint = sample.endpoint, | ||||
|             **sample.payload, | ||||
|             ) | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| @fixture(name='mock_get_request_client', autouse=True) | ||||
| def fixture_mock_get_request_client(sample): | ||||
|     with patch('scwrypts.http.discord.client.get_request_client') as mock: | ||||
|         mock.return_value = lambda method, endpoint, **kwargs: sample.response | ||||
|         yield mock | ||||
|  | ||||
| @fixture(name='_mock_getenv') | ||||
| def fixture_mock_getenv(sample): | ||||
|     with patch('scwrypts.http.discord.client.getenv',) as mock: | ||||
|         mock.side_effect = lambda name, **kwargs: { | ||||
|                 'DISCORD__BOT_TOKEN': sample.bot_token, | ||||
|                 }[name] | ||||
|         yield mock | ||||
|  | ||||
| @fixture(name='_mock_getenv_optional') | ||||
| def fixture_mock_getenv_optional(): | ||||
|     with patch('scwrypts.http.discord.client.getenv',) as mock: | ||||
|         mock.side_effect = lambda name, **kwargs: None | ||||
|         yield mock | ||||
							
								
								
									
										91
									
								
								py/lib/scwrypts/http/discord/test_send_message.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								py/lib/scwrypts/http/discord/test_send_message.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | ||||
| from unittest.mock import patch | ||||
|  | ||||
| from pytest import fixture, raises | ||||
|  | ||||
| from .send_message import send_message | ||||
|  | ||||
|  | ||||
| def test_discord_send_message(sample, mock_request, _mock_getenv): | ||||
|     expected = get_default_called_with(sample) | ||||
|  | ||||
|     assert send_message(sample.content) == sample.response | ||||
|     mock_request.assert_called_once_with(**expected) | ||||
|  | ||||
| def test_discord_send_message_without_username(sample, mock_request, _mock_getenv): | ||||
|     sample.username = None | ||||
|     expected = get_default_called_with(sample) | ||||
|     del expected['json']['username'] | ||||
|  | ||||
|     assert send_message(sample.content) == sample.response | ||||
|     mock_request.assert_called_once_with(**expected) | ||||
|  | ||||
| def test_discord_send_message_without_avatar_url(sample, mock_request, _mock_getenv): | ||||
|     sample.avatar_url = None | ||||
|     expected = get_default_called_with(sample) | ||||
|     del expected['json']['avatar_url'] | ||||
|  | ||||
|     assert send_message(sample.content) == sample.response | ||||
|     mock_request.assert_called_once_with(**expected) | ||||
|  | ||||
| def test_discord_send_message_to_channel_id(sample, mock_request, _mock_getenv): | ||||
|     sample.webhook = None | ||||
|     expected = get_default_called_with(sample) | ||||
|     expected['endpoint'] = f'channels/{sample.channel_id}/messages' | ||||
|  | ||||
|     assert send_message(sample.content) == sample.response | ||||
|     mock_request.assert_called_once_with(**expected) | ||||
|  | ||||
| def test_discord_send_message_without_content_header(sample, mock_request, _mock_getenv): | ||||
|     sample.content_header = None | ||||
|     expected = get_default_called_with(sample) | ||||
|     expected['json']['content'] = f'{sample.content}{sample.content_footer}' | ||||
|  | ||||
|     assert send_message(sample.content) == sample.response | ||||
|     mock_request.assert_called_once_with(**expected) | ||||
|  | ||||
| def test_discord_send_message_without_content_footer(sample, mock_request, _mock_getenv): | ||||
|     sample.content_footer = None | ||||
|     expected = get_default_called_with(sample) | ||||
|     expected['json']['content'] = f'{sample.content_header}{sample.content}' | ||||
|  | ||||
|     assert send_message(sample.content) == sample.response | ||||
|     mock_request.assert_called_once_with(**expected) | ||||
|  | ||||
| def test_discord_send_message_error(sample, mock_request, _mock_getenv): | ||||
|     with raises(ValueError): | ||||
|         sample.webhook = None | ||||
|         sample.channel_id = None | ||||
|  | ||||
|         send_message(sample.content) | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| def get_default_called_with(sample): | ||||
|     return { | ||||
|             'method': 'POST', | ||||
|             'endpoint': f'webhooks/{sample.webhook}', | ||||
|             'json': { | ||||
|                 'content': f'{sample.content_header}{sample.content}{sample.content_footer}', | ||||
|                 'username': sample.username, | ||||
|                 'avatar_url': sample.avatar_url, | ||||
|                 }, | ||||
|             } | ||||
|  | ||||
| @fixture(name='mock_request', autouse=True) | ||||
| def fixture_mock_request(sample): | ||||
|     with patch('scwrypts.http.discord.send_message.request') as mock: | ||||
|         mock.return_value = sample.response | ||||
|         yield mock | ||||
|  | ||||
| @fixture(name='_mock_getenv') | ||||
| def fixture_mock_getenv(sample): | ||||
|     with patch('scwrypts.http.discord.send_message.getenv',) as mock: | ||||
|         mock.side_effect = lambda name, **kwargs: { | ||||
|                 'DISCORD__DEFAULT_USERNAME': sample.username, | ||||
|                 'DISCORD__DEFAULT_AVATAR_URL': sample.avatar_url, | ||||
|                 'DISCORD__DEFAULT_WEBHOOK': sample.webhook, | ||||
|                 'DISCORD__DEFAULT_CHANNEL_ID': sample.channel_id, | ||||
|                 'DISCORD__CONTENT_HEADER': sample.content_header, | ||||
|                 'DISCORD__CONTENT_FOOTER': sample.content_footer, | ||||
|                 }[name] | ||||
|         yield mock | ||||
							
								
								
									
										14
									
								
								py/lib/scwrypts/http/linear/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								py/lib/scwrypts/http/linear/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| ''' | ||||
| basic scwrypts.http client for linear | ||||
|  | ||||
| configured by setting the LINEAR__API_TOKEN option in the | ||||
| scwrypts environment | ||||
| ''' | ||||
|  | ||||
| __all__ = [ | ||||
|         'request', | ||||
|         'graphql', | ||||
|         ] | ||||
|  | ||||
| from .client import request | ||||
| from .graphql import graphql | ||||
							
								
								
									
										12
									
								
								py/lib/scwrypts/http/linear/client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								py/lib/scwrypts/http/linear/client.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| from scwrypts.env import getenv | ||||
|  | ||||
| from .. import get_request_client | ||||
|  | ||||
|  | ||||
| def request(method, endpoint, **kwargs): | ||||
|     return get_request_client( | ||||
|             base_url = 'https://api.linear.app', | ||||
|             headers = { | ||||
|                 'Authorization': f'bearer {getenv("LINEAR__API_TOKEN")}', | ||||
|                 }, | ||||
|             )(method, endpoint, **kwargs) | ||||
							
								
								
									
										19
									
								
								py/lib/scwrypts/http/linear/conftest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								py/lib/scwrypts/http/linear/conftest.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| from string import ascii_letters, digits | ||||
| from types import SimpleNamespace | ||||
|  | ||||
| from pytest import fixture | ||||
|  | ||||
| from scwrypts.test import generate | ||||
| from scwrypts.test.character_set import uri | ||||
| from ..conftest import options, get_request_client_sample_data | ||||
|  | ||||
| @fixture(name='sample') | ||||
| def fixture_sample(): | ||||
|     return SimpleNamespace( | ||||
|             **{ | ||||
|                 **get_request_client_sample_data(), | ||||
|                 'base_url': 'https://api.linear.app', | ||||
|                 }, | ||||
|             api_token = generate(str, options | {'character_set': uri}), | ||||
|             query     = generate(str, options), | ||||
|         ) | ||||
							
								
								
									
										5
									
								
								py/lib/scwrypts/http/linear/graphql.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								py/lib/scwrypts/http/linear/graphql.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| from .client import request | ||||
|  | ||||
|  | ||||
| def graphql(query): | ||||
|     return request('POST', 'graphql', json={'query': query}) | ||||
							
								
								
									
										42
									
								
								py/lib/scwrypts/http/linear/test_client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								py/lib/scwrypts/http/linear/test_client.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| from unittest.mock import patch | ||||
|  | ||||
| from pytest import fixture | ||||
|  | ||||
| from .client import request | ||||
|  | ||||
|  | ||||
| def test_discord_request(sample, _response): | ||||
|     assert _response == sample.response | ||||
|  | ||||
| def test_discord_request_client_setup(sample, mock_get_request_client, _response): | ||||
|     mock_get_request_client.assert_called_once_with( | ||||
|             base_url = sample.base_url, | ||||
|             headers = { 'Authorization': f'bearer {sample.api_token}' }, | ||||
|             ) | ||||
|  | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| @fixture(name='_response') | ||||
| def fixture_response(sample): | ||||
|     return request( | ||||
|             method   = sample.method, | ||||
|             endpoint = sample.endpoint, | ||||
|             **sample.payload, | ||||
|             ) | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| @fixture(name='mock_get_request_client', autouse=True) | ||||
| def fixture_mock_get_request_client(sample): | ||||
|     with patch('scwrypts.http.linear.client.get_request_client') as mock: | ||||
|         mock.return_value = lambda method, endpoint, **kwargs: sample.response | ||||
|         yield mock | ||||
|  | ||||
| @fixture(name='mock_getenv', autouse=True) | ||||
| def fixture_mock_getenv(sample): | ||||
|     with patch('scwrypts.http.linear.client.getenv',) as mock: | ||||
|         mock.side_effect = lambda name, **kwargs: { | ||||
|                 'LINEAR__API_TOKEN': sample.api_token, | ||||
|                 }[name] | ||||
|         yield mock | ||||
							
								
								
									
										35
									
								
								py/lib/scwrypts/http/linear/test_graphql.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								py/lib/scwrypts/http/linear/test_graphql.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| from unittest.mock import patch | ||||
|  | ||||
| from pytest import fixture | ||||
|  | ||||
| from .graphql import graphql | ||||
|  | ||||
|  | ||||
| def test_directus_graphql(sample, _response, _mock_request): | ||||
|     assert _response == sample.response | ||||
|  | ||||
| def test_directus_graphql_request_payload(sample, _response, _mock_request): | ||||
|     _mock_request.assert_called_once_with( | ||||
|             'POST', | ||||
|             'graphql', | ||||
|             json = {'query': sample.query}, | ||||
|             ) | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| @fixture(name='_response') | ||||
| def fixture_response(sample, _mock_request): | ||||
|     return graphql(sample.query) | ||||
|  | ||||
|  | ||||
| @fixture(name='_response_system') | ||||
| def fixture_response_system(sample, _mock_request): | ||||
|     return graphql(sample.query) | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| @fixture(name='_mock_request') | ||||
| def fixture_mock_request(sample): | ||||
|     with patch('scwrypts.http.linear.graphql.request') as mock: | ||||
|         mock.return_value = sample.response | ||||
|         yield mock | ||||
							
								
								
									
										55
									
								
								py/lib/scwrypts/http/test_client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								py/lib/scwrypts/http/test_client.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| from unittest.mock import patch | ||||
|  | ||||
| from pytest import fixture | ||||
|  | ||||
| from .client import get_request_client | ||||
|  | ||||
|  | ||||
| def test_request_client(sample, _response_basic): | ||||
|     assert _response_basic == sample.response | ||||
|  | ||||
| def test_request_client_forwards_default_headers(sample, mock_request, _response_basic): | ||||
|     mock_request.assert_called_once_with( | ||||
|             method = sample.method, | ||||
|             url = f'{sample.base_url}/{sample.endpoint}', | ||||
|             headers = sample.headers, | ||||
|             ) | ||||
|  | ||||
| def test_get_request_client_payload(sample, _response_payload): | ||||
|     assert _response_payload == sample.response | ||||
|  | ||||
| def test_request_client_forwards_payload_headers(sample, mock_request, _response_payload): | ||||
|     assert mock_request.call_args.kwargs['headers'] == sample.headers | sample.payload_headers | ||||
|  | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| @fixture(name='mock_request', autouse=True) | ||||
| def fixture_mock_request(sample): | ||||
|     with patch('scwrypts.http.client.request') as mock: | ||||
|         mock.return_value = sample.response | ||||
|         yield mock | ||||
|  | ||||
| @fixture(name='request_client', autouse=True) | ||||
| def fixture_request_client(sample): | ||||
|     return get_request_client(sample.base_url, sample.headers) | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| @fixture(name='_response_basic') | ||||
| def fixture_response_basic(sample, request_client): | ||||
|     return request_client( | ||||
|             method   = sample.method, | ||||
|             endpoint = sample.endpoint, | ||||
|             ) | ||||
|  | ||||
| @fixture(name='_response_payload') | ||||
| def fixture_response_payload(sample, request_client): | ||||
|     return request_client( | ||||
|             method   = sample.method, | ||||
|             endpoint = sample.endpoint, | ||||
|             **{ | ||||
|                 **sample.payload, | ||||
|                 'headers': sample.payload_headers, | ||||
|                 }, | ||||
|             ) | ||||
							
								
								
									
										1
									
								
								py/lib/scwrypts/io/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								py/lib/scwrypts/io/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| from .combined_io_stream import get_combined_stream, add_io_arguments | ||||
| @@ -2,7 +2,47 @@ from contextlib import contextmanager | ||||
| from pathlib import Path | ||||
| from sys import stdin, stdout, stderr | ||||
| 
 | ||||
| from py.lib.scwrypts.getenv import getenv | ||||
| from scwrypts.env import getenv | ||||
| 
 | ||||
| 
 | ||||
| @contextmanager | ||||
| def get_combined_stream(input_file=None, output_file=None): | ||||
|     ''' | ||||
|     context manager to open an "input_file" and "output_file" | ||||
| 
 | ||||
|     But the "files" can be pipe-streams, stdin/stdout, or even | ||||
|     actual files! Helpful when trying to write CLI scwrypts | ||||
|     which would like to accept all kinds of input and output | ||||
|     configurations. | ||||
|     ''' | ||||
|     with get_stream(input_file, 'r') as input_stream, get_stream(output_file, 'w+') as output_stream: | ||||
|         yield CombinedStream(input_stream, output_stream) | ||||
| 
 | ||||
| def add_io_arguments(parser, allow_input=True, allow_output=True): | ||||
|     ''' | ||||
|     slap these puppies onto your argparse.ArgumentParser to | ||||
|     allow easy use of the get_combined_stream at the command line | ||||
|     ''' | ||||
|     if allow_input: | ||||
|         parser.add_argument( | ||||
|                 '-i', '--input-file', | ||||
|                 dest     = 'input_file', | ||||
|                 default  = None, | ||||
|                 help     = 'path to input file; omit for stdin', | ||||
|                 required = False, | ||||
|                 ) | ||||
| 
 | ||||
|     if allow_output: | ||||
|         parser.add_argument( | ||||
|                 '-o', '--output-file', | ||||
|                 dest     = 'output_file', | ||||
|                 default  = None, | ||||
|                 help     = 'path to output file; omit for stdout', | ||||
|                 required = False, | ||||
|                 ) | ||||
| 
 | ||||
| 
 | ||||
| ##################################################################### | ||||
| 
 | ||||
| 
 | ||||
| @contextmanager | ||||
| @@ -34,32 +74,6 @@ def get_stream(filename=None, mode='r', encoding='utf-8', verbose=False, **kwarg | ||||
|             stdout.flush() | ||||
| 
 | ||||
| 
 | ||||
| def add_io_arguments(parser, toggle_input=True, toggle_output=True): | ||||
|     if toggle_input: | ||||
|         parser.add_argument( | ||||
|                 '-i', '--input-file', | ||||
|                 dest     = 'input_file', | ||||
|                 default  = None, | ||||
|                 help     = 'path to input file; omit for stdin', | ||||
|                 required = False, | ||||
|                 ) | ||||
| 
 | ||||
|     if toggle_output: | ||||
|         parser.add_argument( | ||||
|                 '-o', '--output-file', | ||||
|                 dest     = 'output_file', | ||||
|                 default  = None, | ||||
|                 help     = 'path to output file; omit for stdout', | ||||
|                 required = False, | ||||
|                 ) | ||||
| 
 | ||||
| 
 | ||||
| @contextmanager | ||||
| def get_combined_stream(input_file=None, output_file=None): | ||||
|     with get_stream(input_file, 'r') as input_stream, get_stream(output_file, 'w+') as output_stream: | ||||
|         yield CombinedStream(input_stream, output_stream) | ||||
| 
 | ||||
| 
 | ||||
| class CombinedStream: | ||||
|     def __init__(self, input_stream, output_stream): | ||||
|         self.input = input_stream | ||||
							
								
								
									
										1
									
								
								py/lib/scwrypts/redis/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								py/lib/scwrypts/redis/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| from .client import get_client | ||||
| @@ -1,6 +1,6 @@ | ||||
| from redis import StrictRedis | ||||
| 
 | ||||
| from py.lib.scwrypts import getenv | ||||
| from scwrypts.env import getenv | ||||
| 
 | ||||
| CLIENT = None | ||||
| 
 | ||||
| @@ -1,22 +0,0 @@ | ||||
| from os import getenv | ||||
| from pathlib import Path | ||||
| from subprocess import run as subprocess_run | ||||
|  | ||||
|  | ||||
| def run(scwrypt_name, *args): | ||||
|     DEPTH = int(getenv('SUBSCWRYPT', '0')) | ||||
|     DEPTH += 1 | ||||
|  | ||||
|     SCWRYPTS_EXE = Path(__file__).parents[3] / 'scwrypts' | ||||
|     ARGS = ' '.join([str(x) for x in args]) | ||||
|     print(f'SUBSCWRYPT={DEPTH} {SCWRYPTS_EXE} {scwrypt_name} -- {ARGS}') | ||||
|  | ||||
|     print(f'\n {"--"*DEPTH} ({DEPTH}) BEGIN SUBSCWRYPT : {Path(scwrypt_name).name}') | ||||
|     subprocess_run( | ||||
|         f'SUBSCWRYPT={DEPTH} {SCWRYPTS_EXE} {scwrypt_name} -- {ARGS}', | ||||
|         shell=True, | ||||
|         executable='/bin/zsh', | ||||
|         check=False, | ||||
|         ) | ||||
|  | ||||
|     print(f' {"--"*DEPTH} ({DEPTH}) END SUBSCWRYPT   : {Path(scwrypt_name).name}\n') | ||||
							
								
								
									
										23
									
								
								py/lib/scwrypts/scwrypts/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								py/lib/scwrypts/scwrypts/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| ''' | ||||
| scwrypts meta-configuration | ||||
|  | ||||
| provides a helpful three ways to run "scwrypts" | ||||
|  | ||||
|  'scwrypts' is an agnostic, top-level executor allowing any scwrypt to be called from python workflows | ||||
|  | ||||
|  'execute' is the default context set-up for python-based scwrypts | ||||
|  | ||||
|  'interactive' is a context set-up for interactive, python-based scwrypts | ||||
|    after execution, you are dropped in a bpython shell with all the variables | ||||
|    configured during main() execution | ||||
| ''' | ||||
|  | ||||
| __all__ = [ | ||||
|         'scwrypts', | ||||
|         'execute', | ||||
|         'interactive', | ||||
|         ] | ||||
|  | ||||
| from .scwrypts import scwrypts | ||||
| from .execute import execute | ||||
| from .interactive import interactive | ||||
							
								
								
									
										26
									
								
								py/lib/scwrypts/scwrypts/exceptions.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								py/lib/scwrypts/scwrypts/exceptions.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| from argparse import ArgumentError | ||||
|  | ||||
|  | ||||
| class MissingVariableError(EnvironmentError): | ||||
|     def init(self, name): | ||||
|         super().__init__(f'Missing required environment variable "{name}"') | ||||
|  | ||||
|  | ||||
| class MissingFlagAndEnvironmentVariableError(EnvironmentError, ArgumentError): | ||||
|     def __init__(self, flags, env_var): | ||||
|         super().__init__(f'must provide at least one of : {{ flags: {flags} OR {env_var} }}') | ||||
|  | ||||
|  | ||||
| class MissingScwryptsExecutableError(EnvironmentError): | ||||
|     def __init__(self): | ||||
|         super().__init__(f'scwrypts must be installed and available on your PATH') | ||||
|  | ||||
|  | ||||
| class BadScwryptsLookupError(ValueError): | ||||
|     def __init__(self): | ||||
|         super().__init__('must provide name/group/type or scwrypt lookup patterns') | ||||
|  | ||||
|  | ||||
| class MissingScwryptsGroupOrTypeError(ValueError): | ||||
|     def __init__(self, group, _type): | ||||
|         super().__init__(f'missing required group or type (group={group} | type={_type}') | ||||
| @@ -1,9 +1,12 @@ | ||||
| from argparse import ArgumentParser, ArgumentDefaultsHelpFormatter | ||||
| 
 | ||||
| from py.lib.scwrypts.io import get_combined_stream, add_io_arguments | ||||
| from scwrypts.io import get_combined_stream, add_io_arguments | ||||
| 
 | ||||
| 
 | ||||
| def execute(main, description=None, parse_args=None, toggle_input=True, toggle_output=True): | ||||
| def execute(main, description=None, parse_args=None, allow_input=True, allow_output=True): | ||||
|     ''' | ||||
|     API to initiate a python-based scwrypt | ||||
|     ''' | ||||
|     if parse_args is None: | ||||
|         parse_args = [] | ||||
| 
 | ||||
| @@ -12,7 +15,7 @@ def execute(main, description=None, parse_args=None, toggle_input=True, toggle_o | ||||
|             formatter_class = ArgumentDefaultsHelpFormatter, | ||||
|             ) | ||||
| 
 | ||||
|     add_io_arguments(parser, toggle_input, toggle_output) | ||||
|     add_io_arguments(parser, allow_input, allow_output) | ||||
| 
 | ||||
|     for a in parse_args: | ||||
|         parser.add_argument(*a[0], **a[1]) | ||||
| @@ -2,6 +2,9 @@ from bpython import embed | ||||
| 
 | ||||
| 
 | ||||
| def interactive(variable_descriptions): | ||||
|     ''' | ||||
|     main() decorator to drop to interactive python environment upon completion | ||||
|     ''' | ||||
|     def outer(function): | ||||
| 
 | ||||
|         def inner(*args, **kwargs): | ||||
							
								
								
									
										48
									
								
								py/lib/scwrypts/scwrypts/scwrypts.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								py/lib/scwrypts/scwrypts/scwrypts.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| from os import getenv | ||||
| from shutil import which | ||||
| from subprocess import run | ||||
|  | ||||
| from .exceptions import MissingScwryptsExecutableError, BadScwryptsLookupError, MissingScwryptsGroupOrTypeError | ||||
|  | ||||
|  | ||||
| def scwrypts(*args, patterns=None, name=None, group=None, _type=None, log_level=None): | ||||
|     ''' | ||||
|     top-level scwrypts invoker from python | ||||
|  | ||||
|     - patterns allows for pattern-based scwrypt lookup | ||||
|     - name/group/type allos for precise-match lookup | ||||
|  | ||||
|     *args should be a list of strings and is forwarded to the | ||||
|     invoked scwrypt | ||||
|  | ||||
|     see 'scwrypts --help' for more information | ||||
|     ''' | ||||
|     executable = which('scwrypts') | ||||
|     if executable is None: | ||||
|         raise MissingScwryptsExecutableError() | ||||
|  | ||||
|     if patterns is None and name is None: | ||||
|         raise BadScwryptsLookupError() | ||||
|  | ||||
|     pre_args = [] | ||||
|  | ||||
|     if name is None: | ||||
|         pre_args += patterns | ||||
|     else: | ||||
|         pre_args += ['--name', name, '--group', group, '--type', _type] | ||||
|         if group is None or _type is None: | ||||
|             raise MissingScwryptsGroupOrTypeError(group, _type) | ||||
|  | ||||
|     if log_level is not None: | ||||
|         pre_args += ['--log-level', log_level] | ||||
|  | ||||
|     depth = getenv('SUBSCWRYPT', '') | ||||
|     if depth != '': | ||||
|         depth = int(depth) + 1 | ||||
|  | ||||
|     return run( | ||||
|         f'SUBSCWRYPT={depth} {executable} {pre_args} -- {" ".join(args)}', | ||||
|         shell=True, | ||||
|         executable='/bin/zsh', | ||||
|         check=False, | ||||
|         ) | ||||
							
								
								
									
										10
									
								
								py/lib/scwrypts/test/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								py/lib/scwrypts/test/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| ''' | ||||
| automated testing utilties, but primarily a random data generator | ||||
| ''' | ||||
| __all__ = [ | ||||
|         'generate', | ||||
|         ] | ||||
|  | ||||
| from .generate import generate | ||||
|  | ||||
| from .character_set import * | ||||
							
								
								
									
										13
									
								
								py/lib/scwrypts/test/character_set.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								py/lib/scwrypts/test/character_set.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| ''' | ||||
| string constants typically used for randomly generated data | ||||
|  | ||||
| the 'string' standard library already contains many character sets, | ||||
| but not these :) | ||||
| ''' | ||||
|  | ||||
| __all__ = [ | ||||
|         'uri', | ||||
|         ] | ||||
|  | ||||
|  | ||||
| uri = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-._~:/?#[]@!$&\'()*+,;=' | ||||
							
								
								
									
										10
									
								
								py/lib/scwrypts/test/exceptions.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								py/lib/scwrypts/test/exceptions.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| class GeneratorError(Exception): | ||||
|     pass | ||||
|  | ||||
| class NoDataTypeError(GeneratorError, ValueError): | ||||
|     def __init__(self): | ||||
|         super().__init__('must provide at least one data type (either "data_type" or "data_types")') | ||||
|  | ||||
| class BadGeneratorTypeError(GeneratorError, ValueError): | ||||
|     def __init__(self, data_type): | ||||
|         super().__init__(f'no generator exists for data type "{data_type}"') | ||||
							
								
								
									
										372
									
								
								py/lib/scwrypts/test/generate.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										372
									
								
								py/lib/scwrypts/test/generate.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,372 @@ | ||||
| from csv import writer, QUOTE_NONNUMERIC | ||||
| from io import StringIO | ||||
| from json import dumps, loads | ||||
| from random import randint, uniform, choice | ||||
| from re import sub | ||||
| from string import printable | ||||
| from typing import Hashable, Callable | ||||
| from uuid import uuid4 | ||||
|  | ||||
| from requests import Response, status_codes | ||||
| from yaml import safe_dump | ||||
|  | ||||
| from .exceptions import NoDataTypeError, BadGeneratorTypeError | ||||
|  | ||||
|  | ||||
| DEFAULT_OPTIONS = { | ||||
|         'data_types': None, | ||||
|         'minimum':  0, | ||||
|         'maximum': 64, | ||||
|         'depth': 1, | ||||
|         'character_set': None, | ||||
|         'bool_nullable': False, | ||||
|         'str_length': None, | ||||
|         'str_length_minimum': 0, | ||||
|         'str_length_maximum': 64, | ||||
|         'uuid_output_type': 'uuid',  # str or 'uuid' | ||||
|         'list_length': 8, | ||||
|         'set_length': 8, | ||||
|         'dict_length': 8, | ||||
|         'csv_bool_nullable': True, | ||||
|         'csv_columns': None, | ||||
|         'csv_columns_minimum': 1, | ||||
|         'csv_columns_maximum': 16, | ||||
|         'csv_rows': None, | ||||
|         'csv_rows_minimum': 2, | ||||
|         'csv_rows_maximum': 16, | ||||
|         'csv_output_type': 'stringio',  # str or 'stringio' | ||||
|         'json_initial_type': dict,  # typically dict or list | ||||
|         'json_bool_nullable': True, | ||||
|         'json_output_type': 'stringio',  # str or 'stringio' | ||||
|         'yaml_initial_type': dict,  # typically dict or list | ||||
|         'yaml_bool_nullable': True, | ||||
|         'yaml_use_default_flow_style': False, | ||||
|         'yaml_output_type': 'stringio',  # str or 'stringio' | ||||
|         'requests_response_status_code': status_codes.codes[200], | ||||
|         } | ||||
|  | ||||
| def generate(data_type=None, options=None): | ||||
|     ''' | ||||
|     generate random data with the call of a function | ||||
|         use data_type to generate a single value | ||||
|  | ||||
|         use options to set generation options (key = type, value = kwargs) | ||||
|  | ||||
|         use options.data_types and omit data_type to generate a random type | ||||
|     ''' | ||||
|     if options is None: | ||||
|         options = {} | ||||
|  | ||||
|     options = DEFAULT_OPTIONS | options | ||||
|  | ||||
|     if data_type is None: | ||||
|         if options['data_types'] is None or len(options['data_types']) == 0: | ||||
|             raise NoDataTypeError() | ||||
|  | ||||
|         return generate( | ||||
|                 data_type=choice(list(options['data_types'])), | ||||
|                 options=options, | ||||
|                 ) | ||||
|  | ||||
|     if not isinstance(data_type, str): | ||||
|         data_type = data_type.__name__ | ||||
|  | ||||
|     if data_type not in Generator.get_supported_data_types(): | ||||
|         raise BadGeneratorTypeError(data_type) | ||||
|  | ||||
|     return getattr(Generator, f'_{data_type}')(options) | ||||
|  | ||||
| ##################################################################### | ||||
|  | ||||
| SUPPORTED_DATA_TYPES = None | ||||
|  | ||||
| class Generator: | ||||
|  | ||||
|     @classmethod | ||||
|     def get_supported_data_types(cls): | ||||
|         global SUPPORTED_DATA_TYPES  # pylint: disable=global-statement | ||||
|  | ||||
|         if SUPPORTED_DATA_TYPES is None: | ||||
|             SUPPORTED_DATA_TYPES = { | ||||
|                     sub('^_', '', data_type) | ||||
|                     for data_type, method in Generator.__dict__.items() | ||||
|                     if isinstance(method, staticmethod) | ||||
|                     } | ||||
|  | ||||
|         return SUPPORTED_DATA_TYPES | ||||
|  | ||||
|     ##################################################################### | ||||
|  | ||||
|     @classmethod | ||||
|     def filter_data_types(cls, options, filters=None): | ||||
|         ''' | ||||
|         returns an options dict with appropriately filtered data_types | ||||
|  | ||||
|         if data_types are not yet defined, starts with all supported data_types | ||||
|         ''' | ||||
|         if options['data_types'] is None: | ||||
|             options['data_types'] = Generator.get_supported_data_types() | ||||
|  | ||||
|         if filters is None or len(filters) == 0: | ||||
|             return options | ||||
|  | ||||
|         return { | ||||
|                 **options, | ||||
|                 'data_types': set(filter( | ||||
|                     lambda data_type: all(( f(data_type, options) for f in filters )), | ||||
|                     options['data_types'], | ||||
|                     )), | ||||
|                 } | ||||
|  | ||||
|     class Filters: | ||||
|         @staticmethod | ||||
|         def hashable(data_type, _options): | ||||
|             if isinstance(data_type, Callable): | ||||
|                 return isinstance(data_type(), Hashable) | ||||
|             if not isinstance(data_type, str): | ||||
|                 data_type = data_type.__name__ | ||||
|             return data_type in { 'bool', 'int', 'float', 'chr', 'str', 'uuid' } | ||||
|  | ||||
|         @staticmethod | ||||
|         def filelike(data_type, _options): | ||||
|             return data_type in { 'csv', 'json', 'yaml' } | ||||
|  | ||||
|         @staticmethod | ||||
|         def complex(data_type, _options): | ||||
|             return data_type in { 'requests_Response' } | ||||
|  | ||||
|         @staticmethod | ||||
|         def basic(data_type, options): | ||||
|             return all([ | ||||
|                 not Generator.Filters.filelike(data_type, options), | ||||
|                 not Generator.Filters.complex(data_type, options), | ||||
|                 ]) | ||||
|  | ||||
|         @staticmethod | ||||
|         def pythonset(data_type, _options): | ||||
|             if not isinstance(data_type, str): | ||||
|                 data_type = data_type.__name__ | ||||
|             return data_type == 'set' | ||||
|  | ||||
|         @staticmethod | ||||
|         def csvsafe(data_type, options): | ||||
|             options['depth'] = max(1, options['depth']) | ||||
|             return all([ | ||||
|                 Generator.Filters.basic(data_type, options), | ||||
|                 not Generator.Filters.pythonset(data_type, options), | ||||
|                 ]) | ||||
|  | ||||
|         @staticmethod | ||||
|         def jsonsafe(data_type, options): | ||||
|             return all([ | ||||
|                 Generator.Filters.basic(data_type, options), | ||||
|                 not Generator.Filters.pythonset(data_type, options), | ||||
|                 ]) | ||||
|  | ||||
|         @staticmethod | ||||
|         def yamlsafe(data_type, options): | ||||
|             return all([ | ||||
|                 Generator.Filters.basic(data_type, options), | ||||
|                 not Generator.Filters.pythonset(data_type, options), | ||||
|                 ]) | ||||
|  | ||||
|     ##################################################################### | ||||
|  | ||||
|     @classmethod | ||||
|     def get_option_with_range(cls, options, option_key, data_type=int): | ||||
|         ''' | ||||
|         typically an integer range, allows both: | ||||
|             - setting a fixed configuration (e.g. 'str_length') | ||||
|             - allowing a configuration range (e.g. 'str_length_minimum' and 'str_length_maximum') | ||||
|         ''' | ||||
|         fixed = options.get(option_key, None) | ||||
|         if fixed is not None: | ||||
|             return fixed | ||||
|  | ||||
|         return generate(data_type, { | ||||
|             'minimum': options[f'{option_key}_minimum'], | ||||
|             'maximum': options[f'{option_key}_maximum'], | ||||
|             }) | ||||
|  | ||||
|     ##################################################################### | ||||
|  | ||||
|  | ||||
|     @staticmethod | ||||
|     def _bool(options): | ||||
|         return choice([True, False, None]) if options['bool_nullable'] else choice([True, False]) | ||||
|  | ||||
|  | ||||
|     @staticmethod | ||||
|     def _int(options): | ||||
|         return randint(options['minimum'], options['maximum']) | ||||
|  | ||||
|  | ||||
|     @staticmethod | ||||
|     def _float(options): | ||||
|         return uniform(options['minimum'], options['maximum']) | ||||
|  | ||||
|  | ||||
|     @staticmethod | ||||
|     def _chr(options): | ||||
|         character_set = options['character_set'] | ||||
|         return choice(character_set) if character_set is not None else chr(randint(0,65536)) | ||||
|  | ||||
|  | ||||
|     @staticmethod | ||||
|     def _str(options): | ||||
|         return ''.join(( | ||||
|             generate(chr, options) | ||||
|             for _ in range(Generator.get_option_with_range(options, 'str_length')) | ||||
|             )) | ||||
|  | ||||
|  | ||||
|     @staticmethod | ||||
|     def _uuid(options): | ||||
|         ''' | ||||
|         creates a UUID object or a str containing a uuid (v4) | ||||
|         ''' | ||||
|         uuid = uuid4() | ||||
|         return str(uuid) if options['uuid_output_type'] == str else uuid | ||||
|  | ||||
|  | ||||
|     @staticmethod | ||||
|     def _list(options): | ||||
|         if options['depth'] <= 0: | ||||
|             return [] | ||||
|  | ||||
|         options['depth'] -= 1 | ||||
|         options = Generator.filter_data_types(options, [ | ||||
|             Generator.Filters.basic, | ||||
|             ]) | ||||
|  | ||||
|         return [ generate(None, {**options}) for _ in range(options['list_length']) ] | ||||
|  | ||||
|  | ||||
|     @staticmethod | ||||
|     def _set(options): | ||||
|         if options['depth'] <= 0: | ||||
|             return set() | ||||
|  | ||||
|         options['depth'] -= 1 | ||||
|         options = Generator.filter_data_types(options, [ | ||||
|             Generator.Filters.hashable, | ||||
|             ]) | ||||
|  | ||||
|         return { generate(None, options) for _ in range(options['set_length']) } | ||||
|  | ||||
|  | ||||
|     @staticmethod | ||||
|     def _dict(options): | ||||
|         if options['depth'] <= 0: | ||||
|             return {} | ||||
|  | ||||
|         options['depth'] -= 1 | ||||
|         options = Generator.filter_data_types(options, [ | ||||
|             Generator.Filters.basic, | ||||
|             ]) | ||||
|  | ||||
|         key_options = Generator.filter_data_types(options, [ | ||||
|             Generator.Filters.hashable, | ||||
|             ]) | ||||
|  | ||||
|         if len(options['data_types']) == 0 or len(key_options['data_types']) == 0: | ||||
|             return {} | ||||
|  | ||||
|         return { | ||||
|                 generate(None, key_options): generate(None, options) | ||||
|                 for _ in range(options['dict_length']) | ||||
|                 } | ||||
|  | ||||
|  | ||||
|     @staticmethod | ||||
|     def _csv(options): | ||||
|         ''' | ||||
|         creates a StringIO object containing csv data | ||||
|         ''' | ||||
|         if options['character_set'] is None: | ||||
|             options['character_set'] = printable | ||||
|  | ||||
|         options['bool_nullable'] = options['csv_bool_nullable'] | ||||
|         options = Generator.filter_data_types(options, [ | ||||
|             Generator.Filters.csvsafe, | ||||
|             ]) | ||||
|  | ||||
|         columns = Generator.get_option_with_range(options, 'csv_columns') | ||||
|         rows    = Generator.get_option_with_range(options, 'csv_rows') | ||||
|  | ||||
|  | ||||
|         csv = StringIO() | ||||
|         csv_writer = writer(csv, quoting=QUOTE_NONNUMERIC) | ||||
|  | ||||
|         options['list_length'] = columns | ||||
|  | ||||
|         [  # pylint: disable=expression-not-assigned | ||||
|                 csv_writer.writerow(generate(list, options)) | ||||
|                 for _ in range(rows) | ||||
|                 ] | ||||
|  | ||||
|         csv.seek(0) | ||||
|         return csv.getvalue() if options['csv_output_type'] == str else csv | ||||
|  | ||||
|  | ||||
|     @staticmethod | ||||
|     def _json(options): | ||||
|         ''' | ||||
|         creates a StringIO object or str containing json data | ||||
|         ''' | ||||
|  | ||||
|         if options['character_set'] is None: | ||||
|             options['character_set'] = printable | ||||
|  | ||||
|         options['bool_nullable'] = options['json_bool_nullable'] | ||||
|         options['uuid_output_type'] = str | ||||
|         options = Generator.filter_data_types(options, [ | ||||
|             Generator.Filters.jsonsafe, | ||||
|             ]) | ||||
|  | ||||
|         json = dumps(generate( | ||||
|             options['json_initial_type'], | ||||
|             {**options}, | ||||
|             )) | ||||
|  | ||||
|         return json if options['json_output_type'] == str else StringIO(json) | ||||
|  | ||||
|  | ||||
|     @staticmethod | ||||
|     def _yaml(options): | ||||
|         ''' | ||||
|         creates a StringIO object or str containing yaml data | ||||
|         ''' | ||||
|         if options['character_set'] is None: | ||||
|             options['character_set'] = printable | ||||
|  | ||||
|         options['bool_nullable'] = options['yaml_bool_nullable'] | ||||
|         options['uuid_output_type'] = str | ||||
|         options = Generator.filter_data_types(options, [ | ||||
|             Generator.Filters.yamlsafe, | ||||
|             ]) | ||||
|  | ||||
|         yaml = StringIO() | ||||
|         safe_dump( | ||||
|                 generate(options['yaml_initial_type'], {**options}), | ||||
|                 yaml, | ||||
|                 default_flow_style=options['yaml_use_default_flow_style'], | ||||
|                 ) | ||||
|         yaml.seek(0) | ||||
|  | ||||
|         return yaml.getvalue() if options['yaml_output_type'] == str else yaml | ||||
|  | ||||
|     @staticmethod | ||||
|     def _requests_Response(options): | ||||
|         ''' | ||||
|         creates a requests.Response-like object containing json data | ||||
|         ''' | ||||
|  | ||||
|         options['json_output_type'] = str | ||||
|  | ||||
|         response = Response() | ||||
|         response.status_code = options['requests_response_status_code'] | ||||
|         json = loads(generate('json', options)) | ||||
|         response.json = lambda: json | ||||
|  | ||||
|         return response | ||||
							
								
								
									
										54
									
								
								py/lib/scwrypts/test/test_generate.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								py/lib/scwrypts/test/test_generate.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | ||||
| from os import getenv | ||||
| from pprint import pprint | ||||
| from random import randint | ||||
|  | ||||
| from .generate import generate, Generator | ||||
|  | ||||
| ITERATIONS = int( | ||||
|         getenv( | ||||
|             'PYTEST_ITERATIONS__scwrypts__test__generator', | ||||
|             getenv('PYTEST_ITERATIONS', '99'),  # CI should use at least 999 | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
| FILE_LIKE_DATA_TYPES = { 'csv', 'json', 'yaml' } | ||||
|  | ||||
| def test_generate():  # generators should be quick and "just work" (no Exceptions) | ||||
|     print() | ||||
|     for data_type in Generator.get_supported_data_types(): | ||||
|         print(f'------- {data_type} -------') | ||||
|         sample = generate(data_type) | ||||
|         pprint(sample.getvalue() if data_type in {'csv', 'json', 'yaml'} else sample) | ||||
|         for _ in range(ITERATIONS): | ||||
|             generate(data_type) | ||||
|  | ||||
|  | ||||
| def test_generate_depth_deep(): | ||||
|     for data_type in Generator.get_supported_data_types(): | ||||
|         generate(data_type, {'depth': 4}) | ||||
|  | ||||
| def test_generate_depth_shallow(): | ||||
|     for data_type in Generator.get_supported_data_types(): | ||||
|         generate(data_type, {'depth': randint(-999, 0)}) | ||||
|  | ||||
|  | ||||
| def test_generate_range_all(): | ||||
|     for data_type in Generator.get_supported_data_types(): | ||||
|         generate(data_type, {'minimum': -99, 'maximum': 99}) | ||||
|  | ||||
| def test_generate_range_positive(): | ||||
|     for data_type in Generator.get_supported_data_types(): | ||||
|         generate(data_type, {'minimum':   1, 'maximum': 99}) | ||||
|  | ||||
| def test_generate_range_zero(): | ||||
|     for data_type in Generator.get_supported_data_types(): | ||||
|         generate(data_type, {'minimum':   3, 'maximum':  3}) | ||||
|  | ||||
| def test_generate_range_negative(): | ||||
|     for data_type in Generator.get_supported_data_types(): | ||||
|         generate(data_type, {'minimum': -99, 'maximum': -1}) | ||||
|  | ||||
|  | ||||
| def test_generate_bool_nullable(): | ||||
|     for data_type in Generator.get_supported_data_types(): | ||||
|         generate(data_type, {'bool_nullable': True}) | ||||
							
								
								
									
										13
									
								
								py/lib/scwrypts/twilio/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								py/lib/scwrypts/twilio/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| ''' | ||||
| loads the twilio.rest.Client by referencing TWILIO__API_KEY, | ||||
| TWILIO__API_SECRET, and TWILIO__ACCOUNT_SID in your scwrypts | ||||
| environment | ||||
| ''' | ||||
|  | ||||
| __all__ = [ | ||||
|         'get_client', | ||||
|         'send_sms', | ||||
|         ] | ||||
|  | ||||
| from .client import get_client | ||||
| from .send_sms import send_sms | ||||
| @@ -1,6 +1,6 @@ | ||||
| from twilio.rest import Client | ||||
| 
 | ||||
| from py.lib.scwrypts import getenv | ||||
| from scwrypts.env import getenv | ||||
| 
 | ||||
| CLIENT = None | ||||
| 
 | ||||
| @@ -1,7 +1,7 @@ | ||||
| from json import dumps | ||||
| from time import sleep | ||||
| 
 | ||||
| from py.lib.twilio.client import get_client | ||||
| from .client import get_client | ||||
| 
 | ||||
| 
 | ||||
| def send_sms(to, from_, body, max_char_count=300, stream=None): | ||||
| @@ -1,2 +0,0 @@ | ||||
| from py.lib.twilio.client import get_client | ||||
| from py.lib.twilio.send_sms import send_sms | ||||
		Reference in New Issue
	
	Block a user