abstract-utilities 0.2.2.476__py3-none-any.whl → 0.2.2.540__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 abstract-utilities might be problematic. Click here for more details.

Files changed (190) hide show
  1. abstract_utilities/__init__.py +6 -10
  2. abstract_utilities/circular_import_finder.py +222 -0
  3. abstract_utilities/circular_import_finder2.py +118 -0
  4. abstract_utilities/class_utils/__init__.py +7 -0
  5. abstract_utilities/class_utils/abstract_classes.py +74 -0
  6. abstract_utilities/class_utils/caller_utils.py +35 -0
  7. abstract_utilities/class_utils/class_utils.py +109 -0
  8. abstract_utilities/class_utils/function_utils.py +153 -0
  9. abstract_utilities/class_utils/global_utils.py +56 -0
  10. abstract_utilities/class_utils/imports/__init__.py +2 -0
  11. abstract_utilities/class_utils/imports/imports.py +2 -0
  12. abstract_utilities/class_utils/imports/utils.py +40 -0
  13. abstract_utilities/class_utils/module_utils.py +63 -0
  14. abstract_utilities/directory_utils/__init__.py +2 -0
  15. abstract_utilities/directory_utils/directory_utils.py +94 -0
  16. abstract_utilities/directory_utils/imports/__init__.py +2 -0
  17. abstract_utilities/directory_utils/imports/imports.py +1 -0
  18. abstract_utilities/directory_utils/imports/module_imports.py +2 -0
  19. abstract_utilities/directory_utils/name_utils.py +43 -0
  20. abstract_utilities/directory_utils/size_utils.py +57 -0
  21. abstract_utilities/directory_utils/src/__init__.py +4 -0
  22. abstract_utilities/directory_utils/src/directory_utils.py +92 -0
  23. abstract_utilities/directory_utils/src/name_utils.py +43 -0
  24. abstract_utilities/directory_utils/src/size_utils.py +57 -0
  25. abstract_utilities/directory_utils/src/utils.py +116 -0
  26. abstract_utilities/directory_utils/utils.py +116 -0
  27. abstract_utilities/env_utils/imports/imports.py +5 -3
  28. abstract_utilities/error_utils/__init__.py +2 -0
  29. abstract_utilities/error_utils/error_utils.py +25 -0
  30. abstract_utilities/error_utils/imports/__init__.py +2 -0
  31. abstract_utilities/error_utils/imports/imports.py +1 -0
  32. abstract_utilities/error_utils/imports/module_imports.py +1 -0
  33. abstract_utilities/file_utils/__init__.py +1 -2
  34. abstract_utilities/file_utils/file_utils/imports/module_imports.py +2 -1
  35. abstract_utilities/file_utils/file_utils/type_checks.py +34 -24
  36. abstract_utilities/file_utils/imports/constants.py +84 -4
  37. abstract_utilities/file_utils/imports/imports.py +3 -18
  38. abstract_utilities/file_utils/imports/module_imports.py +2 -7
  39. abstract_utilities/file_utils/module_imports.py +12 -0
  40. abstract_utilities/file_utils/src/__init__.py +8 -0
  41. abstract_utilities/file_utils/src/file_filters.py +177 -0
  42. abstract_utilities/file_utils/src/file_reader.py +544 -0
  43. abstract_utilities/file_utils/src/file_utils.py +156 -0
  44. abstract_utilities/file_utils/src/filter_params.py +197 -0
  45. abstract_utilities/file_utils/src/find_collect.py +266 -0
  46. abstract_utilities/file_utils/src/initFunctionsGen.py +280 -0
  47. abstract_utilities/file_utils/src/map_utils.py +29 -0
  48. abstract_utilities/file_utils/src/pdf_utils.py +300 -0
  49. abstract_utilities/file_utils/src/type_checks.py +91 -0
  50. abstract_utilities/file_utils (2)/__init__.py +2 -0
  51. abstract_utilities/file_utils (2)/imports/__init__.py +2 -0
  52. abstract_utilities/file_utils (2)/imports/constants.py +118 -0
  53. abstract_utilities/file_utils (2)/imports/imports/__init__.py +3 -0
  54. abstract_utilities/file_utils (2)/imports/imports/constants.py +119 -0
  55. abstract_utilities/file_utils (2)/imports/imports/imports.py +46 -0
  56. abstract_utilities/file_utils (2)/imports/imports/module_imports.py +8 -0
  57. abstract_utilities/file_utils (2)/imports/utils/__init__.py +3 -0
  58. abstract_utilities/file_utils (2)/imports/utils/classes.py +379 -0
  59. abstract_utilities/file_utils (2)/imports/utils/clean_imps.py +155 -0
  60. abstract_utilities/file_utils (2)/imports/utils/filter_utils.py +341 -0
  61. abstract_utilities/file_utils (2)/src/__init__.py +8 -0
  62. abstract_utilities/file_utils (2)/src/file_filters.py +155 -0
  63. abstract_utilities/file_utils (2)/src/file_reader.py +604 -0
  64. abstract_utilities/file_utils (2)/src/find_collect.py +258 -0
  65. abstract_utilities/file_utils (2)/src/initFunctionsGen.py +286 -0
  66. abstract_utilities/file_utils (2)/src/map_utils.py +28 -0
  67. abstract_utilities/file_utils (2)/src/pdf_utils.py +300 -0
  68. abstract_utilities/hash_utils/__init__.py +2 -0
  69. abstract_utilities/hash_utils/hash_utils.py +5 -0
  70. abstract_utilities/hash_utils/imports/__init__.py +2 -0
  71. abstract_utilities/hash_utils/imports/imports.py +1 -0
  72. abstract_utilities/hash_utils/imports/module_imports.py +0 -0
  73. abstract_utilities/history_utils/__init__.py +2 -0
  74. abstract_utilities/history_utils/history_utils.py +37 -0
  75. abstract_utilities/history_utils/imports/__init__.py +2 -0
  76. abstract_utilities/history_utils/imports/imports.py +1 -0
  77. abstract_utilities/history_utils/imports/module_imports.py +0 -0
  78. abstract_utilities/import_utils/__init__.py +2 -0
  79. abstract_utilities/import_utils/imports/__init__.py +4 -0
  80. abstract_utilities/import_utils/imports/constants.py +2 -0
  81. abstract_utilities/import_utils/imports/imports.py +4 -0
  82. abstract_utilities/import_utils/imports/module_imports.py +6 -0
  83. abstract_utilities/import_utils/imports/utils.py +30 -0
  84. abstract_utilities/import_utils/src/__init__.py +7 -0
  85. abstract_utilities/import_utils/src/clean_imports.py +147 -0
  86. abstract_utilities/import_utils/src/dot_utils.py +69 -0
  87. abstract_utilities/import_utils/src/extract_utils.py +42 -0
  88. abstract_utilities/import_utils/src/import_functions.py +46 -0
  89. abstract_utilities/import_utils/src/import_utils.py +299 -0
  90. abstract_utilities/import_utils/src/package_utils/__init__.py +139 -0
  91. abstract_utilities/import_utils/src/package_utils/context_utils.py +27 -0
  92. abstract_utilities/import_utils/src/package_utils/import_collectors.py +53 -0
  93. abstract_utilities/import_utils/src/package_utils/path_utils.py +28 -0
  94. abstract_utilities/import_utils/src/package_utils/safe_import.py +27 -0
  95. abstract_utilities/import_utils/src/package_utils.py +140 -0
  96. abstract_utilities/import_utils/src/package_utilss/__init__.py +139 -0
  97. abstract_utilities/import_utils/src/package_utilss/context_utils.py +27 -0
  98. abstract_utilities/import_utils/src/package_utilss/import_collectors.py +53 -0
  99. abstract_utilities/import_utils/src/package_utilss/path_utils.py +28 -0
  100. abstract_utilities/import_utils/src/package_utilss/safe_import.py +27 -0
  101. abstract_utilities/import_utils/src/pkg_utils.py +140 -0
  102. abstract_utilities/import_utils/src/sysroot_utils.py +57 -0
  103. abstract_utilities/imports.py +18 -0
  104. abstract_utilities/json_utils/__init__.py +2 -0
  105. abstract_utilities/json_utils/imports/__init__.py +2 -0
  106. abstract_utilities/json_utils/imports/imports.py +2 -0
  107. abstract_utilities/json_utils/imports/module_imports.py +5 -0
  108. abstract_utilities/json_utils/json_utils.py +743 -0
  109. abstract_utilities/list_utils/__init__.py +2 -0
  110. abstract_utilities/list_utils/imports/__init__.py +2 -0
  111. abstract_utilities/list_utils/imports/imports.py +1 -0
  112. abstract_utilities/list_utils/imports/module_imports.py +0 -0
  113. abstract_utilities/list_utils/list_utils.py +199 -0
  114. abstract_utilities/log_utils/__init__.py +5 -0
  115. abstract_utilities/log_utils/abstractLogManager.py +64 -0
  116. abstract_utilities/log_utils/call_response.py +68 -0
  117. abstract_utilities/log_utils/imports/__init__.py +2 -0
  118. abstract_utilities/log_utils/imports/imports.py +7 -0
  119. abstract_utilities/log_utils/imports/module_imports.py +2 -0
  120. abstract_utilities/log_utils/log_file.py +58 -0
  121. abstract_utilities/log_utils/logger_callable.py +49 -0
  122. abstract_utilities/math_utils/__init__.py +2 -0
  123. abstract_utilities/math_utils/imports/__init__.py +2 -0
  124. abstract_utilities/math_utils/imports/imports.py +2 -0
  125. abstract_utilities/math_utils/imports/module_imports.py +1 -0
  126. abstract_utilities/math_utils/math_utils.py +208 -0
  127. abstract_utilities/parse_utils/__init__.py +2 -0
  128. abstract_utilities/parse_utils/imports/__init__.py +3 -0
  129. abstract_utilities/parse_utils/imports/constants.py +10 -0
  130. abstract_utilities/parse_utils/imports/imports.py +2 -0
  131. abstract_utilities/parse_utils/imports/module_imports.py +4 -0
  132. abstract_utilities/parse_utils/parse_utils.py +516 -0
  133. abstract_utilities/path_utils/__init__.py +2 -0
  134. abstract_utilities/path_utils/imports/__init__.py +3 -0
  135. abstract_utilities/path_utils/imports/imports.py +1 -0
  136. abstract_utilities/path_utils/imports/module_imports.py +8 -0
  137. abstract_utilities/path_utils/path_utils.py +251 -0
  138. abstract_utilities/path_utils.py +95 -14
  139. abstract_utilities/read_write_utils/__init__.py +1 -0
  140. abstract_utilities/read_write_utils/imports/__init__.py +2 -0
  141. abstract_utilities/read_write_utils/imports/imports.py +2 -0
  142. abstract_utilities/read_write_utils/imports/module_imports.py +5 -0
  143. abstract_utilities/read_write_utils/read_write_utils.py +338 -0
  144. abstract_utilities/read_write_utils.py +142 -20
  145. abstract_utilities/safe_utils/__init__.py +2 -0
  146. abstract_utilities/safe_utils/imports/__init__.py +3 -0
  147. abstract_utilities/safe_utils/imports/imports.py +2 -0
  148. abstract_utilities/safe_utils/imports/module_imports.py +2 -0
  149. abstract_utilities/safe_utils/safe_utils.py +136 -0
  150. abstract_utilities/ssh_utils/__init__.py +3 -1
  151. abstract_utilities/ssh_utils/classes.py +0 -1
  152. abstract_utilities/ssh_utils/cmd_utils.py +207 -0
  153. abstract_utilities/ssh_utils/imports/__init__.py +3 -0
  154. abstract_utilities/ssh_utils/imports/imports.py +5 -0
  155. abstract_utilities/ssh_utils/imports/module_imports.py +6 -0
  156. abstract_utilities/ssh_utils/imports/utils.py +189 -0
  157. abstract_utilities/ssh_utils/pexpect_utils.py +11 -18
  158. abstract_utilities/ssh_utils/type_checks.py +92 -0
  159. abstract_utilities/string_utils/__init__.py +4 -0
  160. abstract_utilities/string_utils/clean_utils.py +28 -0
  161. abstract_utilities/string_utils/eat_utils.py +103 -0
  162. abstract_utilities/string_utils/imports/__init__.py +3 -0
  163. abstract_utilities/string_utils/imports/imports.py +2 -0
  164. abstract_utilities/string_utils/imports/module_imports.py +2 -0
  165. abstract_utilities/string_utils/imports/utils.py +81 -0
  166. abstract_utilities/string_utils/replace_utils.py +27 -0
  167. abstract_utilities/string_utils.py +4 -1
  168. abstract_utilities/thread_utils/__init__.py +2 -0
  169. abstract_utilities/thread_utils/imports/__init__.py +2 -0
  170. abstract_utilities/thread_utils/imports/imports.py +2 -0
  171. abstract_utilities/thread_utils/imports/module_imports.py +2 -0
  172. abstract_utilities/thread_utils/thread_utils.py +140 -0
  173. abstract_utilities/time_utils/__init__.py +2 -0
  174. abstract_utilities/time_utils/imports/__init__.py +2 -0
  175. abstract_utilities/time_utils/imports/imports.py +3 -0
  176. abstract_utilities/time_utils/imports/module_imports.py +1 -0
  177. abstract_utilities/time_utils/time_utils.py +392 -0
  178. abstract_utilities/type_utils/__init__.py +3 -0
  179. abstract_utilities/type_utils/alpha_utils.py +59 -0
  180. abstract_utilities/type_utils/imports/__init__.py +2 -0
  181. abstract_utilities/type_utils/imports/imports.py +4 -0
  182. abstract_utilities/type_utils/imports/module_imports.py +1 -0
  183. abstract_utilities/type_utils/num_utils.py +19 -0
  184. abstract_utilities/type_utils/type_utils.py +981 -0
  185. {abstract_utilities-0.2.2.476.dist-info → abstract_utilities-0.2.2.540.dist-info}/METADATA +1 -1
  186. abstract_utilities-0.2.2.540.dist-info/RECORD +263 -0
  187. imports/__init__.py +36 -0
  188. abstract_utilities-0.2.2.476.dist-info/RECORD +0 -92
  189. {abstract_utilities-0.2.2.476.dist-info → abstract_utilities-0.2.2.540.dist-info}/WHEEL +0 -0
  190. {abstract_utilities-0.2.2.476.dist-info → abstract_utilities-0.2.2.540.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,197 @@
1
+
2
+ from ..imports import *
3
+ import re
4
+ def combine_params(*values,typ=None):
5
+ nu_values = None
6
+ for value in values:
7
+ if value is not None:
8
+ typ = typ or type(value)
9
+ if nu_values is None:
10
+ nu_values = typ()
11
+
12
+ if typ is set:
13
+ nu_values = nu_values | typ(value)
14
+ if typ is list:
15
+ nu_values += typ(value)
16
+ return nu_values
17
+ def get_safe_kwargs(canonical_map, **kwargs):
18
+ # Lowercase all keys for safety
19
+ canonical_map = canonical_map or CANONICAL_MAP
20
+ norm_kwargs = {k.lower(): v for k, v in kwargs.items() if v is not None}
21
+
22
+ # Inverse lookup: alias → canonical key
23
+ alias_lookup = {
24
+ alias: canon
25
+ for canon, aliases in canonical_map.items()
26
+ if aliases
27
+ for alias in aliases
28
+ }
29
+
30
+ # Preserve correctly named keys
31
+ safe_kwargs = {k: v for k, v in norm_kwargs.items() if k in canonical_map}
32
+
33
+ for k, v in norm_kwargs.items():
34
+ if k in alias_lookup:
35
+ canonical_key = alias_lookup[k]
36
+ prev = safe_kwargs.get(canonical_key)
37
+ if prev is None:
38
+ safe_kwargs[canonical_key] = v
39
+ else:
40
+ # merge intelligently if both exist
41
+ if isinstance(prev, (set, list)) and isinstance(v, (set, list)):
42
+ safe_kwargs[canonical_key] = list(set(prev) | set(v))
43
+ else:
44
+ safe_kwargs[canonical_key] = v # overwrite for non-iterables
45
+
46
+ # fill defaults if missing
47
+ for canon in canonical_map:
48
+ safe_kwargs.setdefault(canon, None)
49
+
50
+ return safe_kwargs
51
+
52
+ def create_canonical_map(*args,canonical_map=None):
53
+ keys = [arg for arg in args if arg]
54
+ if not keys:
55
+ return CANONICAL_MAP
56
+ canonical_map = canonical_map or CANONICAL_MAP
57
+
58
+ return {key:canonical_map.get(key) for key in keys}
59
+ def get_safe_canonical_kwargs(*args,canonical_map=None,**kwargs):
60
+ canonical_map = canonical_map or create_canonical_map(*args)
61
+
62
+ return get_safe_kwargs(canonical_map=canonical_map,**kwargs)
63
+ def get_dir_filter_kwargs(**kwargs):
64
+ canonical_map = create_canonical_map("directories")
65
+ return get_safe_kwargs(canonical_map=canonical_map,**kwargs)
66
+ def get_file_filter_kwargs(**kwargs):
67
+ """
68
+ Normalize arbitrary keyword arguments for file scanning configuration.
69
+
70
+ Examples:
71
+ - 'excluded_ext' or 'unallowed_exts' → 'exclude_exts'
72
+ - 'include_dirs' or 'allow_dir' → 'allowed_dirs'
73
+ - 'excludePattern' or 'excluded_patterns' → 'exclude_patterns'
74
+ - 'allowed_type' or 'include_types' → 'allowed_types'
75
+ """
76
+ # Canonical keys and aliases
77
+ canonical_keys =["allowed_exts","exclude_exts","allowed_types","exclude_types","allowed_dirs","exclude_dirs","allowed_patterns","exclude_patterns"]
78
+
79
+ return get_safe_canonical_kwargs(*canonical_keys,**kwargs)
80
+ # -------------------------
81
+ # Utility functions
82
+ # -------------------------
83
+
84
+ def _normalize_listlike(value, typ=list, sep=','):
85
+ """Normalize comma-separated or iterable values into the desired type."""
86
+ if value in [True, None, False]:
87
+ return value
88
+ if isinstance(value, str):
89
+ value = [v.strip() for v in value.split(sep) if v.strip()]
90
+ return typ(value)
91
+
92
+ def ensure_exts(exts):
93
+ if exts in [True, None, False]:
94
+ return exts
95
+ out = []
96
+ for ext in _normalize_listlike(exts, list):
97
+ if not ext.startswith('.'):
98
+ ext = f".{ext}"
99
+ out.append(ext)
100
+ return set(out)
101
+
102
+ def ensure_patterns(patterns):
103
+ """Normalize pattern list and ensure they are valid globs."""
104
+ if patterns in [True, None, False]:
105
+ return patterns
106
+ patterns = _normalize_listlike(patterns, list)
107
+ out = []
108
+ for pattern in patterns:
109
+ if not pattern:
110
+ continue
111
+ if '*' not in pattern and '?' not in pattern:
112
+ # Implicitly make it a prefix match
113
+ if pattern.startswith('.') or pattern.startswith('~'):
114
+ pattern = f"*{pattern}"
115
+ else:
116
+ pattern = f"{pattern}*"
117
+ out.append(pattern)
118
+ return out
119
+
120
+
121
+ def _get_default_modular(value, default, add=False, typ=set):
122
+ """Merge user and default values intelligently."""
123
+ if value == None:
124
+ value = add
125
+ if value in [True]:
126
+ return default
127
+ if value is False:
128
+ return value
129
+ if add:
130
+ return combine_params(value,default,typ=None)
131
+
132
+ return typ(value)
133
+
134
+ # -------------------------
135
+ # Default derivation logic
136
+ # -------------------------
137
+ def _get_default_modular(value, default, add=None, typ=set):
138
+ """Merge user and default values intelligently."""
139
+ add = add or False
140
+ if value == None:
141
+ value = add
142
+ if value in [True]:
143
+ return default
144
+ if value is False:
145
+ return value
146
+ if add:
147
+ return combine_params(value,default,typ=None)
148
+ return typ(value)
149
+ def derive_all_defaults(**kwargs):
150
+ kwargs = get_safe_canonical_kwargs(**kwargs)
151
+ add = kwargs.get("add",False)
152
+ nu_defaults = {}
153
+ for key,values in DEFAULT_CANONICAL_MAP.items():
154
+ default = values.get("default")
155
+ typ = values.get("type")
156
+ key_value = kwargs.get(key)
157
+ if key in DEFAULT_ALLOWED_EXCLUDE_MAP:
158
+
159
+ if key.endswith('exts'):
160
+ input_value = ensure_exts(key_value)
161
+ if key.endswith('patterns'):
162
+ input_value = ensure_patterns(key_value)
163
+ else:
164
+ input_value = _normalize_listlike(key_value, typ)
165
+ nu_defaults[key] = _get_default_modular(input_value, default, add, typ)
166
+ else:
167
+ value = default if key_value is None else key_value
168
+ if typ == list:
169
+ value = make_list(value)
170
+ elif typ == bool:
171
+ value = bool(value)
172
+ nu_defaults[key] = value
173
+
174
+ return nu_defaults
175
+ # -------------------------
176
+ # Default derivation logic
177
+ # -------------------------
178
+ def derive_file_defaults(**kwargs):
179
+ kwargs = derive_all_defaults(**kwargs)
180
+ add = kwargs.get("add",False)
181
+ nu_defaults = {}
182
+ for key,values in DEFAULT_ALLOWED_EXCLUDE_MAP.items():
183
+ default = values.get("default")
184
+ typ = values.get("type")
185
+ key_value = kwargs.get(key)
186
+ if key.endswith('exts'):
187
+ input_value = ensure_exts(key_value)
188
+ if key.endswith('patterns'):
189
+ input_value = ensure_patterns(key_value)
190
+ else:
191
+ input_value = _normalize_listlike(key_value, typ)
192
+ nu_defaults[key] = _get_default_modular(input_value, default, add, typ)
193
+ return nu_defaults
194
+
195
+ def define_defaults(**kwargs):
196
+ defaults = derive_file_defaults(**kwargs)
197
+ return ScanConfig(**defaults)
@@ -0,0 +1,266 @@
1
+ from ..imports import *
2
+ from .file_filters import *
3
+ from .filter_params import *
4
+ from pathlib import Path
5
+ from typing import Optional, List, Set
6
+
7
+ def get_proper_type_str(string):
8
+ if not string:
9
+ return None
10
+ string_lower = string.lower()
11
+ items = {
12
+ "d":["dir","dirs","directory","directories","d","dirname"],
13
+ "f":["file","filepath","file_path","files","filepaths","file_paths","f"]
14
+ }
15
+ for key,values in items.items():
16
+ if string_lower in values:
17
+ return key
18
+ init = string_lower[0] if len(string_lower)>0 else None
19
+ if init in items:
20
+ return init
21
+ def check_path_type(
22
+ path: str,
23
+ user: Optional[str] = None,
24
+ host: Optional[str] = None,
25
+ user_as_host: Optional[str] = None,
26
+ use_shell: bool = False
27
+ ) -> Literal["file", "directory", "missing", "unknown"]:
28
+ """
29
+ Determine whether a given path is a file, directory, or missing.
30
+ Works locally or remotely (via SSH).
31
+
32
+ Args:
33
+ path: The path to check.
34
+ user, host, user_as_host: SSH parameters if remote.
35
+ use_shell: Force shell test instead of Python os.path.
36
+ Returns:
37
+ One of: 'file', 'directory', 'missing', or 'unknown'
38
+ """
39
+
40
+ # --- remote check if user/host is given ---
41
+ if user_as_host or (user and host):
42
+ remote_target = user_as_host or f"{user}@{host}"
43
+ cmd = f"if [ -f '{path}' ]; then echo file; elif [ -d '{path}' ]; then echo directory; else echo missing; fi"
44
+ try:
45
+ result = subprocess.check_output(
46
+ ["ssh", remote_target, cmd],
47
+ stderr=subprocess.DEVNULL,
48
+ text=True,
49
+ timeout=5
50
+ ).strip()
51
+ return result if result in ("file", "directory", "missing") else "unknown"
52
+ except Exception:
53
+ return "unknown"
54
+
55
+ # --- local check ---
56
+ if not use_shell:
57
+ if os.path.isfile(path):
58
+ return "file"
59
+ elif os.path.isdir(path):
60
+ return "directory"
61
+ elif not os.path.exists(path):
62
+ return "missing"
63
+ return "unknown"
64
+ else:
65
+ # fallback using shell tests (useful for sandboxed contexts)
66
+ cmd = f"if [ -f '{path}' ]; then echo file; elif [ -d '{path}' ]; then echo directory; else echo missing; fi"
67
+ try:
68
+ output = subprocess.check_output(
69
+ cmd, shell=True, stderr=subprocess.DEVNULL, text=True
70
+ ).strip()
71
+ return output if output in ("file", "directory", "missing") else "unknown"
72
+ except Exception:
73
+ return "unknown"
74
+
75
+
76
+
77
+
78
+
79
+ def get_find_cmd(
80
+ *args,
81
+ mindepth: Optional[int] = None,
82
+ maxdepth: Optional[int] = None,
83
+ depth: Optional[int] = None,
84
+ file_type: Optional[str] = None, # 'f' or 'd'
85
+ name: Optional[str] = None,
86
+ size: Optional[str] = None,
87
+ mtime: Optional[str] = None,
88
+ perm: Optional[str] = None,
89
+ user: Optional[str] = None,
90
+ **kwargs
91
+ ) -> str:
92
+ """
93
+ Construct a Unix `find` command string that supports multiple directories.
94
+ Accepts filtering via ScanConfig-compatible kwargs.
95
+ """
96
+ # Normalize inputs into canonical form
97
+ kwargs = get_safe_canonical_kwargs(*args, **kwargs)
98
+ cfg = define_defaults(**kwargs)
99
+
100
+ # Get directory list (may come from args or kwargs)
101
+ directories = [r for r in make_list(kwargs.get("directories")) if r]
102
+ if not directories:
103
+ return []
104
+
105
+ # Build base command for all directories
106
+ dir_expr = " ".join(shlex.quote(d) for d in directories)
107
+ cmd = [f"find {dir_expr}"]
108
+
109
+ # --- depth filters ---
110
+ if depth is not None:
111
+ cmd += [f"-mindepth {depth}", f"-maxdepth {depth}"]
112
+ else:
113
+ if mindepth is not None:
114
+ cmd.append(f"-mindepth {mindepth}")
115
+ if maxdepth is not None:
116
+ cmd.append(f"-maxdepth {maxdepth}")
117
+
118
+ # --- file type ---
119
+ if file_type in ("f", "d"):
120
+ cmd.append(f"-type {file_type}")
121
+
122
+ # --- basic attributes ---
123
+ if name:
124
+ cmd.append(f"-name {shlex.quote(name)}")
125
+ if size:
126
+ cmd.append(f"-size {shlex.quote(size)}")
127
+ if mtime:
128
+ cmd.append(f"-mtime {shlex.quote(mtime)}")
129
+ if perm:
130
+ cmd.append(f"-perm {shlex.quote(perm)}")
131
+ if user:
132
+ cmd.append(f"-user {shlex.quote(user)}")
133
+
134
+ # --- cfg-based filters ---
135
+ if cfg:
136
+ # Allowed extensions
137
+ if cfg.allowed_exts and cfg.allowed_exts != {"*"}:
138
+ ext_expr = " -o ".join(
139
+ [f"-name '*{e}'" for e in cfg.allowed_exts if e]
140
+ )
141
+ cmd.append(f"\\( {ext_expr} \\)")
142
+
143
+ # Excluded extensions
144
+ if cfg.exclude_exts:
145
+ for e in cfg.exclude_exts:
146
+ cmd.append(f"! -name '*{e}'")
147
+
148
+ # Allowed directories
149
+ if cfg.allowed_dirs and cfg.allowed_dirs != ["*"]:
150
+ dir_expr = " -o ".join(
151
+ [f"-path '*{d}*'" for d in cfg.allowed_dirs if d]
152
+ )
153
+ cmd.append(f"\\( {dir_expr} \\)")
154
+
155
+ # Excluded directories
156
+ if cfg.exclude_dirs:
157
+ for d in cfg.exclude_dirs:
158
+ cmd.append(f"! -path '*{d}*'")
159
+
160
+ # Allowed patterns
161
+ if cfg.allowed_patterns and cfg.allowed_patterns != ["*"]:
162
+ pat_expr = " -o ".join(
163
+ [f"-name '{p}'" for p in cfg.allowed_patterns if p]
164
+ )
165
+ cmd.append(f"\\( {pat_expr} \\)")
166
+
167
+ # Excluded patterns
168
+ if cfg.exclude_patterns:
169
+ for p in cfg.exclude_patterns:
170
+ cmd.append(f"! -name '{p}'")
171
+
172
+ # Allowed types (semantic, not `-type`)
173
+ if cfg.allowed_types and cfg.allowed_types != {"*"}:
174
+ type_expr = " -o ".join(
175
+ [f"-path '*{t}*'" for t in cfg.allowed_types if t]
176
+ )
177
+ cmd.append(f"\\( {type_expr} \\)")
178
+
179
+ # Excluded types
180
+ if cfg.exclude_types:
181
+ for t in cfg.exclude_types:
182
+ cmd.append(f"! -path '*{t}*'")
183
+
184
+ return " ".join(cmd)
185
+
186
+ def ensure_directories(*args,**kwargs):
187
+ directories = []
188
+ for arg in args:
189
+ arg_str = str(arg)
190
+ if is_dir(arg_str,**kwargs):
191
+ directories.append(arg_str)
192
+ elif is_file(arg_str,**kwargs):
193
+ dirname = os.path.dirname(arg_str)
194
+ directories.append(dirname)
195
+ safe_directories = get_dir_filter_kwargs(**kwargs)
196
+ directories+= make_list(safe_directories.get('directories',[]))
197
+ return list(set([r for r in directories if r]))
198
+
199
+ def collect_globs(
200
+ *args,
201
+ mindepth: Optional[int] = None,
202
+ maxdepth: Optional[int] = None,
203
+ depth: Optional[int] = None,
204
+ file_type: Optional[str] = None, # "f", "d", or None
205
+ allowed: Optional[Callable[[str], bool]] = None,
206
+ **kwargs
207
+ ) -> List[str] | dict:
208
+ """
209
+ Collect file or directory paths recursively.
210
+
211
+ - If file_type is None → returns {"f": [...], "d": [...]}
212
+ - If file_type is "f" or "d" → returns a list of that type
213
+ - Supports SSH mode via `user_at_host`
214
+ """
215
+ kwargs["directories"] = ensure_directories(*args, **kwargs)
216
+ kwargs= get_safe_canonical_kwargs(**kwargs)
217
+ kwargs["cfg"] = define_defaults(**kwargs)
218
+
219
+ type_strs = {"f":"files","d":"dirs"}
220
+ file_type = get_proper_type_str(file_type)
221
+ file_types = make_list(file_type)
222
+ if file_type == None:
223
+ file_types = ["f","d"]
224
+ return_results = {}
225
+ return_result=[]
226
+ for file_type in file_types:
227
+ type_str = type_strs.get(file_type)
228
+ # Remote path (SSH)
229
+ find_cmd = get_find_cmd(
230
+ mindepth=mindepth,
231
+ maxdepth=maxdepth,
232
+ depth=depth,
233
+ file_type=file_type,
234
+ **{k: v for k, v in kwargs.items() if v},
235
+ )
236
+ result = run_pruned_func(run_cmd,find_cmd,
237
+ **kwargs
238
+
239
+ )
240
+ return_result = [res for res in result.split('\n') if res]
241
+ return_results[type_str]=return_result
242
+ if len(file_types) == 1:
243
+ return return_result
244
+ return return_results
245
+ def get_files_and_dirs(
246
+ *args,
247
+ recursive: bool = True,
248
+ include_files: bool = True,
249
+ **kwargs
250
+ ):
251
+ if recursive == False:
252
+ kwargs['maxdepth']=1
253
+ if include_files == False:
254
+ kwargs['file_type']='d'
255
+ result = collect_globs(*args,**kwargs)
256
+ if include_files == False:
257
+ return result,[]
258
+ dirs = result.get("dirs")
259
+ files = result.get("files")
260
+ return dirs,files
261
+ def collect_filepaths(
262
+ *args,
263
+ **kwargs
264
+ ) -> List[str]:
265
+ kwargs['file_type']='f'
266
+ return collect_globs(*args,**kwargs)