edq-utils 0.1.4__py3-none-any.whl → 0.1.6__py3-none-any.whl

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.

Potentially problematic release.


This version of edq-utils might be problematic. Click here for more details.

edq/__init__.py CHANGED
@@ -2,4 +2,4 @@
2
2
  General Python tools used by several EduLinq projects.
3
3
  """
4
4
 
5
- __version__ = '0.1.4'
5
+ __version__ = '0.1.6'
edq/util/net.py CHANGED
@@ -626,6 +626,22 @@ class HTTPExchange(edq.util.json.DictConverter):
626
626
 
627
627
  return HTTPExchange(**data)
628
628
 
629
+ @typing.runtime_checkable
630
+ class HTTPExchangeComplete(typing.Protocol):
631
+ """
632
+ A function that can be called after a request has been made (and exchange constructed).
633
+ """
634
+
635
+ def __call__(self,
636
+ exchange: HTTPExchange
637
+ ) -> str:
638
+ """
639
+ Called after an HTTP exchange has been completed.
640
+ """
641
+
642
+ _make_request_exchange_complete_func: typing.Union[HTTPExchangeComplete, None] = None # pylint: disable=invalid-name
643
+ """ If not None, call this func after make_request() has created its HTTPExchange. """
644
+
629
645
  def find_open_port(
630
646
  start_port: int = DEFAULT_START_PORT, end_port: int = DEFAULT_END_PORT,
631
647
  wait_time: float = DEFAULT_PORT_SEARCH_WAIT_SEC) -> int:
@@ -671,6 +687,7 @@ def make_request(method: str, url: str,
671
687
  http_exchange_extension: str = DEFAULT_HTTP_EXCHANGE_EXTENSION,
672
688
  add_http_prefix: bool = True,
673
689
  additional_requests_options: typing.Union[typing.Dict[str, typing.Any], None] = None,
690
+ exchange_complete_func: typing.Union[HTTPExchangeComplete, None] = None,
674
691
  **kwargs: typing.Any) -> typing.Tuple[requests.Response, str]:
675
692
  """
676
693
  Make an HTTP request and return the response object and text body.
@@ -713,7 +730,7 @@ def make_request(method: str, url: str,
713
730
  else:
714
731
  options['data'] = data
715
732
 
716
- logging.debug("Making %s request: '%s'.", method, url)
733
+ logging.debug("Making %s request: '%s' (options = %s).", method, url, options)
717
734
  response = requests.request(method, url, **options)
718
735
 
719
736
  body = response.text
@@ -726,9 +743,11 @@ def make_request(method: str, url: str,
726
743
 
727
744
  response.raise_for_status()
728
745
 
729
- if (output_dir is not None):
746
+ exchange = None
747
+ if ((output_dir is not None) or (exchange_complete_func is not None) or (_make_request_exchange_complete_func is not None)):
730
748
  exchange = HTTPExchange.from_response(response, headers_to_skip = headers_to_skip, params_to_skip = params_to_skip)
731
749
 
750
+ if ((output_dir is not None) and (exchange is not None)):
732
751
  path = os.path.abspath(os.path.join(output_dir, *exchange.get_url().split('/')))
733
752
 
734
753
  query = urllib.parse.urlencode(exchange.parameters)
@@ -744,6 +763,12 @@ def make_request(method: str, url: str,
744
763
  edq.util.dirent.mkdir(os.path.dirname(path))
745
764
  edq.util.json.dump_path(exchange, path, indent = 4, sort_keys = False)
746
765
 
766
+ if ((exchange_complete_func is not None) and (exchange is not None)):
767
+ exchange_complete_func(exchange)
768
+
769
+ if ((_make_request_exchange_complete_func is not None) and (exchange is not None)):
770
+ _make_request_exchange_complete_func(exchange) # pylint: disable=not-callable
771
+
747
772
  return response, body
748
773
 
749
774
  def make_get(url: str, **kwargs: typing.Any) -> typing.Tuple[requests.Response, str]:
@@ -862,13 +887,14 @@ def parse_content_dispositions(headers: typing.Union[email.message.Message, typi
862
887
 
863
888
  def parse_query_string(text: str,
864
889
  replace_single_lists: bool = True,
865
- ) -> typing.Dict[str, typing.Any]:
890
+ keep_blank_values: bool = True,
891
+ **kwargs: typing.Any) -> typing.Dict[str, typing.Any]:
866
892
  """
867
893
  Parse a query string (like urllib.parse.parse_qs()), and normalize the result.
868
894
  If specified, lists with single values (as returned from urllib.parse.parse_qs()) will be replaced with the single value.
869
895
  """
870
896
 
871
- results = urllib.parse.parse_qs(text)
897
+ results = urllib.parse.parse_qs(text, keep_blank_values = True)
872
898
  for (key, value) in results.items():
873
899
  if (replace_single_lists and (len(value) == 1)):
874
900
  results[key] = value[0] # type: ignore[assignment]
edq/util/parse.py ADDED
@@ -0,0 +1,33 @@
1
+ import typing
2
+
3
+ BOOL_TRUE_STRINGS: typing.Set[str] = {
4
+ 'true', 't',
5
+ 'yes', 'y',
6
+ '1',
7
+ }
8
+
9
+ BOOL_FALSE_STRINGS: typing.Set[str] = {
10
+ 'false', 'f',
11
+ 'no', 'n',
12
+ '0',
13
+ }
14
+
15
+ def boolean(raw_text: typing.Union[str, bool]) -> bool:
16
+ """
17
+ Parse a boolean from a string using common string representations for true/false.
18
+ This function assumes the entire string is the boolean (not just a part of it).
19
+ If the string is not true or false, then raise an exception.
20
+ """
21
+
22
+ if (isinstance(raw_text, bool)):
23
+ return raw_text
24
+
25
+ text = str(raw_text).lower().strip()
26
+
27
+ if (text in BOOL_TRUE_STRINGS):
28
+ return True
29
+
30
+ if (text in BOOL_FALSE_STRINGS):
31
+ return False
32
+
33
+ raise ValueError(f"Could not convert text to boolean: '{raw_text}'.")
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: edq-utils
3
- Version: 0.1.4
3
+ Version: 0.1.6
4
4
  Summary: Common utilities used by EduLinq Python projects.
5
5
  Author-email: Eriq Augustine <eriq@edulinq.org>
6
6
  License: MIT License
@@ -1,4 +1,4 @@
1
- edq/__init__.py,sha256=ree5ufpagQ7Rc8mHF4u_O4s_xXLJIXNPv8KTxEZhWww,86
1
+ edq/__init__.py,sha256=zzlJP1062nfEQu7AZNJwQURNoVHqDTUn7JuBdLU9Y4M,86
2
2
  edq/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
3
3
  edq/cli/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
4
4
  edq/cli/version.py,sha256=SxarRVD_AVA-nD4pLVMe6ZjSJpMr7h_r3DgYYs42vjE,591
@@ -68,14 +68,15 @@ edq/util/hash.py,sha256=yjXGBCZNvMm49RaPCi9Ygf5FLUpGUrbVU7v8l8hn5Hc,1369
68
68
  edq/util/hash_test.py,sha256=GzvCwgPdzsOn5o63lE8cUurAnj-4arHnYCUcctTnWMQ,2714
69
69
  edq/util/json.py,sha256=nl_cxrlP97RX1BFtys8IT_3IfO0-XvBDQBe6YdrblB4,5936
70
70
  edq/util/json_test.py,sha256=utUVRbw3z42ke4fpRVI294RrFHcMKms8khVYRkISNk4,8009
71
- edq/util/net.py,sha256=UNaFNEyVFxITATHS4-2IcFPkW29WDiIm22fiN7-wUOs,33403
71
+ edq/util/net.py,sha256=rrm7ZmtvMMtvIpQmxRLOg6OUtrWXBlyCmLpI1RyjuMM,34613
72
+ edq/util/parse.py,sha256=zA-GGbY5WF-rfAcWFlnYjDXQaNkxhoyLJ8X81UceCLw,786
72
73
  edq/util/pyimport.py,sha256=26OIuCXELyqtwlooMqDEs4GJQrkrAgxnXNYTlqqtsBY,2852
73
74
  edq/util/pyimport_test.py,sha256=Xno0MIa3yMTfBfoTgjKCIMpr1ZShU6bvo9rBRdecXQU,4202
74
75
  edq/util/reflection.py,sha256=jPcW6h0fwSDYh04O5rUxlgoF7HK6fVQ2mq7DD9qPrEg,972
75
76
  edq/util/time.py,sha256=anoNM_KniARLombv2BnsoHuCzDqMKiDdIzV7RUe2ZOk,2648
76
77
  edq/util/time_test.py,sha256=iQZwzVTVQQ4TdXrLb9MUMCYlKrIe8qyF-hiC9YLTaMo,4610
77
- edq_utils-0.1.4.dist-info/licenses/LICENSE,sha256=MS4iYEl4rOxMoprZuc86iYVoyk4YgaVoMt7WmGvVF8w,1064
78
- edq_utils-0.1.4.dist-info/METADATA,sha256=XlmW8vEdFxmnAUURP0LDyHRAEAH8YYN1dkQ4QJ3NhbM,7535
79
- edq_utils-0.1.4.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
80
- edq_utils-0.1.4.dist-info/top_level.txt,sha256=znBHSj6tgXtcMKrUVtovLli5fIEJCb7d-BMxTLRK4zk,4
81
- edq_utils-0.1.4.dist-info/RECORD,,
78
+ edq_utils-0.1.6.dist-info/licenses/LICENSE,sha256=MS4iYEl4rOxMoprZuc86iYVoyk4YgaVoMt7WmGvVF8w,1064
79
+ edq_utils-0.1.6.dist-info/METADATA,sha256=HD5uwUBWkF0QuE8DO7iuxTVdviSOPyrm-yc_AipERWA,7535
80
+ edq_utils-0.1.6.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
81
+ edq_utils-0.1.6.dist-info/top_level.txt,sha256=znBHSj6tgXtcMKrUVtovLli5fIEJCb7d-BMxTLRK4zk,4
82
+ edq_utils-0.1.6.dist-info/RECORD,,