abstract-utilities 0.2.2.480__py3-none-any.whl → 0.2.2.688__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.
- abstract_utilities/__init__.py +24 -16
- abstract_utilities/circular_import_finder.py +222 -0
- abstract_utilities/circular_import_finder2.py +118 -0
- abstract_utilities/class_utils/__init__.py +7 -0
- abstract_utilities/class_utils/abstract_classes.py +144 -0
- abstract_utilities/class_utils/caller_utils.py +92 -0
- abstract_utilities/class_utils/class_utils.py +109 -0
- abstract_utilities/class_utils/function_utils.py +153 -0
- abstract_utilities/class_utils/global_utils.py +71 -0
- abstract_utilities/class_utils/imports/__init__.py +2 -0
- abstract_utilities/class_utils/imports/imports.py +2 -0
- abstract_utilities/class_utils/imports/utils.py +40 -0
- abstract_utilities/class_utils/module_utils.py +63 -0
- abstract_utilities/directory_utils/__init__.py +2 -0
- abstract_utilities/directory_utils/directory_utils.py +94 -0
- abstract_utilities/directory_utils/imports/__init__.py +2 -0
- abstract_utilities/directory_utils/imports/imports.py +1 -0
- abstract_utilities/directory_utils/imports/module_imports.py +2 -0
- abstract_utilities/directory_utils/name_utils.py +43 -0
- abstract_utilities/directory_utils/size_utils.py +57 -0
- abstract_utilities/directory_utils/src/__init__.py +4 -0
- abstract_utilities/directory_utils/src/directory_utils.py +110 -0
- abstract_utilities/directory_utils/src/name_utils.py +43 -0
- abstract_utilities/directory_utils/src/size_utils.py +57 -0
- abstract_utilities/directory_utils/src/utils.py +116 -0
- abstract_utilities/directory_utils/utils.py +116 -0
- abstract_utilities/env_utils/imports/imports.py +5 -3
- abstract_utilities/error_utils/__init__.py +2 -0
- abstract_utilities/error_utils/error_utils.py +25 -0
- abstract_utilities/error_utils/imports/__init__.py +2 -0
- abstract_utilities/error_utils/imports/imports.py +1 -0
- abstract_utilities/error_utils/imports/module_imports.py +1 -0
- abstract_utilities/file_utils/__init__.py +1 -2
- abstract_utilities/file_utils/file_utils/type_checks.py +2 -1
- abstract_utilities/file_utils/imports/classes.py +59 -55
- abstract_utilities/file_utils/imports/constants.py +84 -4
- abstract_utilities/file_utils/imports/imports.py +2 -21
- abstract_utilities/file_utils/imports/module_imports.py +3 -8
- abstract_utilities/file_utils/module_imports.py +12 -0
- abstract_utilities/file_utils/src/__init__.py +7 -0
- abstract_utilities/file_utils/src/file_filters/__init__.py +1 -0
- abstract_utilities/file_utils/src/file_filters/ensure_utils.py +490 -0
- abstract_utilities/file_utils/src/file_filters/filter_params.py +150 -0
- abstract_utilities/file_utils/src/file_filters/filter_utils.py +78 -0
- abstract_utilities/file_utils/src/file_filters/predicate_utils.py +44 -0
- abstract_utilities/file_utils/src/file_filters.py +177 -0
- abstract_utilities/file_utils/src/file_reader.py +543 -0
- abstract_utilities/file_utils/src/file_utils.py +156 -0
- abstract_utilities/file_utils/src/filter_params.py +197 -0
- abstract_utilities/file_utils/src/find_collect.py +200 -0
- abstract_utilities/file_utils/src/find_content.py +210 -0
- abstract_utilities/file_utils/src/initFunctionsGen.py +293 -0
- abstract_utilities/file_utils/src/initFunctionsGens.py +280 -0
- abstract_utilities/file_utils/src/map_utils.py +29 -0
- abstract_utilities/file_utils/src/pdf_utils.py +300 -0
- abstract_utilities/file_utils/src/reader_utils/__init__.py +4 -0
- abstract_utilities/file_utils/src/reader_utils/directory_reader.py +53 -0
- abstract_utilities/file_utils/src/reader_utils/file_reader.py +543 -0
- abstract_utilities/file_utils/src/reader_utils/file_readers.py +376 -0
- abstract_utilities/file_utils/src/reader_utils/imports.py +18 -0
- abstract_utilities/file_utils/src/reader_utils/pdf_utils.py +300 -0
- abstract_utilities/file_utils/src/type_checks.py +91 -0
- abstract_utilities/file_utils (2)/__init__.py +2 -0
- abstract_utilities/file_utils (2)/imports/__init__.py +2 -0
- abstract_utilities/file_utils (2)/imports/constants.py +118 -0
- abstract_utilities/file_utils (2)/imports/imports/__init__.py +3 -0
- abstract_utilities/file_utils (2)/imports/imports/constants.py +119 -0
- abstract_utilities/file_utils (2)/imports/imports/imports.py +46 -0
- abstract_utilities/file_utils (2)/imports/imports/module_imports.py +8 -0
- abstract_utilities/file_utils (2)/imports/utils/__init__.py +3 -0
- abstract_utilities/file_utils (2)/imports/utils/classes.py +379 -0
- abstract_utilities/file_utils (2)/imports/utils/clean_imps.py +155 -0
- abstract_utilities/file_utils (2)/imports/utils/filter_utils.py +341 -0
- abstract_utilities/file_utils (2)/src/__init__.py +8 -0
- abstract_utilities/file_utils (2)/src/file_filters.py +155 -0
- abstract_utilities/file_utils (2)/src/file_reader.py +604 -0
- abstract_utilities/file_utils (2)/src/find_collect.py +258 -0
- abstract_utilities/file_utils (2)/src/initFunctionsGen.py +286 -0
- abstract_utilities/file_utils (2)/src/map_utils.py +28 -0
- abstract_utilities/file_utils (2)/src/pdf_utils.py +300 -0
- abstract_utilities/hash_utils/__init__.py +2 -0
- abstract_utilities/hash_utils/hash_utils.py +5 -0
- abstract_utilities/hash_utils/imports/__init__.py +2 -0
- abstract_utilities/hash_utils/imports/imports.py +1 -0
- abstract_utilities/hash_utils/imports/module_imports.py +0 -0
- abstract_utilities/history_utils/__init__.py +2 -0
- abstract_utilities/history_utils/history_utils.py +37 -0
- abstract_utilities/history_utils/imports/__init__.py +2 -0
- abstract_utilities/history_utils/imports/imports.py +1 -0
- abstract_utilities/history_utils/imports/module_imports.py +0 -0
- abstract_utilities/import_utils/__init__.py +2 -0
- abstract_utilities/import_utils/circular_import_finder.py +222 -0
- abstract_utilities/import_utils/circular_import_finder2.py +118 -0
- abstract_utilities/import_utils/imports/__init__.py +4 -0
- abstract_utilities/import_utils/imports/constants.py +2 -0
- abstract_utilities/import_utils/imports/imports.py +4 -0
- abstract_utilities/import_utils/imports/init_imports.py +3 -0
- abstract_utilities/import_utils/imports/module_imports.py +9 -0
- abstract_utilities/import_utils/imports/utils.py +30 -0
- abstract_utilities/import_utils/src/__init__.py +8 -0
- abstract_utilities/import_utils/src/clean_imports.py +278 -0
- abstract_utilities/import_utils/src/dot_utils.py +80 -0
- abstract_utilities/import_utils/src/extract_utils.py +46 -0
- abstract_utilities/import_utils/src/import_functions.py +110 -0
- abstract_utilities/import_utils/src/import_utils.py +349 -0
- abstract_utilities/import_utils/src/layze_import_utils/__init__.py +2 -0
- abstract_utilities/import_utils/src/layze_import_utils/lazy_utils.py +41 -0
- abstract_utilities/import_utils/src/layze_import_utils/nullProxy.py +37 -0
- abstract_utilities/import_utils/src/nullProxy.py +30 -0
- abstract_utilities/import_utils/src/package_utils/__init__.py +139 -0
- abstract_utilities/import_utils/src/package_utils/context_utils.py +27 -0
- abstract_utilities/import_utils/src/package_utils/import_collectors.py +53 -0
- abstract_utilities/import_utils/src/package_utils/path_utils.py +28 -0
- abstract_utilities/import_utils/src/package_utils/safe_import.py +27 -0
- abstract_utilities/import_utils/src/package_utils.py +140 -0
- abstract_utilities/import_utils/src/package_utilss/__init__.py +139 -0
- abstract_utilities/import_utils/src/package_utilss/context_utils.py +27 -0
- abstract_utilities/import_utils/src/package_utilss/import_collectors.py +53 -0
- abstract_utilities/import_utils/src/package_utilss/path_utils.py +28 -0
- abstract_utilities/import_utils/src/package_utilss/safe_import.py +27 -0
- abstract_utilities/import_utils/src/pkg_utils.py +194 -0
- abstract_utilities/import_utils/src/sysroot_utils.py +112 -0
- abstract_utilities/imports.py +21 -0
- abstract_utilities/json_utils/__init__.py +2 -0
- abstract_utilities/json_utils/imports/__init__.py +2 -0
- abstract_utilities/json_utils/imports/imports.py +2 -0
- abstract_utilities/json_utils/imports/module_imports.py +5 -0
- abstract_utilities/json_utils/json_utils.py +777 -0
- abstract_utilities/list_utils/__init__.py +2 -0
- abstract_utilities/list_utils/imports/__init__.py +2 -0
- abstract_utilities/list_utils/imports/imports.py +1 -0
- abstract_utilities/list_utils/imports/module_imports.py +0 -0
- abstract_utilities/list_utils/list_utils.py +202 -0
- abstract_utilities/log_utils/__init__.py +5 -0
- abstract_utilities/log_utils/abstractLogManager.py +64 -0
- abstract_utilities/log_utils/call_response.py +68 -0
- abstract_utilities/log_utils/imports/__init__.py +2 -0
- abstract_utilities/log_utils/imports/imports.py +7 -0
- abstract_utilities/log_utils/imports/module_imports.py +2 -0
- abstract_utilities/log_utils/log_file.py +162 -0
- abstract_utilities/log_utils/logger_callable.py +49 -0
- abstract_utilities/math_utils/__init__.py +2 -0
- abstract_utilities/math_utils/imports/__init__.py +2 -0
- abstract_utilities/math_utils/imports/imports.py +2 -0
- abstract_utilities/math_utils/imports/module_imports.py +1 -0
- abstract_utilities/math_utils/math_utils.py +208 -0
- abstract_utilities/parse_utils/__init__.py +2 -0
- abstract_utilities/parse_utils/imports/__init__.py +3 -0
- abstract_utilities/parse_utils/imports/constants.py +10 -0
- abstract_utilities/parse_utils/imports/imports.py +2 -0
- abstract_utilities/parse_utils/imports/module_imports.py +4 -0
- abstract_utilities/parse_utils/parse_utils.py +539 -0
- abstract_utilities/path_utils/__init__.py +2 -0
- abstract_utilities/path_utils/imports/__init__.py +3 -0
- abstract_utilities/path_utils/imports/imports.py +1 -0
- abstract_utilities/path_utils/imports/module_imports.py +8 -0
- abstract_utilities/path_utils/path_utils.py +248 -0
- abstract_utilities/path_utils.py +95 -14
- abstract_utilities/read_write_utils/__init__.py +1 -0
- abstract_utilities/read_write_utils/imports/__init__.py +2 -0
- abstract_utilities/read_write_utils/imports/imports.py +2 -0
- abstract_utilities/read_write_utils/imports/module_imports.py +5 -0
- abstract_utilities/read_write_utils/read_write_utils.py +439 -0
- abstract_utilities/read_write_utils.py +113 -62
- abstract_utilities/safe_utils/__init__.py +2 -0
- abstract_utilities/safe_utils/imports/__init__.py +3 -0
- abstract_utilities/safe_utils/imports/imports.py +2 -0
- abstract_utilities/safe_utils/imports/module_imports.py +2 -0
- abstract_utilities/safe_utils/safe_utils.py +166 -0
- abstract_utilities/ssh_utils/__init__.py +3 -1
- abstract_utilities/ssh_utils/classes.py +0 -1
- abstract_utilities/ssh_utils/cmd_utils.py +207 -0
- abstract_utilities/ssh_utils/imports/__init__.py +3 -0
- abstract_utilities/ssh_utils/imports/imports.py +5 -0
- abstract_utilities/ssh_utils/imports/module_imports.py +6 -0
- abstract_utilities/ssh_utils/imports/utils.py +189 -0
- abstract_utilities/ssh_utils/pexpect_utils.py +11 -18
- abstract_utilities/ssh_utils/type_checks.py +92 -0
- abstract_utilities/string_utils/__init__.py +4 -0
- abstract_utilities/string_utils/clean_utils.py +28 -0
- abstract_utilities/string_utils/eat_utils.py +103 -0
- abstract_utilities/string_utils/imports/__init__.py +3 -0
- abstract_utilities/string_utils/imports/imports.py +2 -0
- abstract_utilities/string_utils/imports/module_imports.py +2 -0
- abstract_utilities/string_utils/imports/utils.py +81 -0
- abstract_utilities/string_utils/replace_utils.py +27 -0
- abstract_utilities/string_utils.py +1 -1
- abstract_utilities/thread_utils/__init__.py +2 -0
- abstract_utilities/thread_utils/imports/__init__.py +2 -0
- abstract_utilities/thread_utils/imports/imports.py +2 -0
- abstract_utilities/thread_utils/imports/module_imports.py +2 -0
- abstract_utilities/thread_utils/thread_utils.py +140 -0
- abstract_utilities/time_utils/__init__.py +2 -0
- abstract_utilities/time_utils/imports/__init__.py +2 -0
- abstract_utilities/time_utils/imports/imports.py +3 -0
- abstract_utilities/time_utils/imports/module_imports.py +1 -0
- abstract_utilities/time_utils/time_utils.py +392 -0
- abstract_utilities/type_utils/__init__.py +7 -0
- abstract_utilities/type_utils/alpha_utils.py +59 -0
- abstract_utilities/type_utils/get_type.py +120 -0
- abstract_utilities/type_utils/imports/__init__.py +3 -0
- abstract_utilities/type_utils/imports/constants.py +134 -0
- abstract_utilities/type_utils/imports/imports.py +4 -0
- abstract_utilities/type_utils/imports/module_imports.py +25 -0
- abstract_utilities/type_utils/is_type.py +455 -0
- abstract_utilities/type_utils/make_type.py +126 -0
- abstract_utilities/type_utils/mime_types.py +68 -0
- abstract_utilities/type_utils/num_utils.py +19 -0
- abstract_utilities/type_utils/type_utils.py +104 -0
- {abstract_utilities-0.2.2.480.dist-info → abstract_utilities-0.2.2.688.dist-info}/METADATA +1 -1
- abstract_utilities-0.2.2.688.dist-info/RECORD +288 -0
- imports/__init__.py +36 -0
- abstract_utilities-0.2.2.480.dist-info/RECORD +0 -92
- {abstract_utilities-0.2.2.480.dist-info → abstract_utilities-0.2.2.688.dist-info}/WHEEL +0 -0
- {abstract_utilities-0.2.2.480.dist-info → abstract_utilities-0.2.2.688.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,200 @@
|
|
|
1
|
+
from ..imports import *
|
|
2
|
+
from .file_filters import *
|
|
3
|
+
from pathlib import Path
|
|
4
|
+
from typing import Optional, List, Set
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
def get_find_cmd(
|
|
12
|
+
*args,
|
|
13
|
+
mindepth: Optional[int] = None,
|
|
14
|
+
maxdepth: Optional[int] = None,
|
|
15
|
+
depth: Optional[int] = None,
|
|
16
|
+
file_type: Optional[str] = None, # 'f' or 'd'
|
|
17
|
+
name: Optional[str] = None,
|
|
18
|
+
size: Optional[str] = None,
|
|
19
|
+
mtime: Optional[str] = None,
|
|
20
|
+
perm: Optional[str] = None,
|
|
21
|
+
user: Optional[str] = None,
|
|
22
|
+
**kwargs
|
|
23
|
+
) -> str:
|
|
24
|
+
"""
|
|
25
|
+
Construct a Unix `find` command string that supports multiple directories.
|
|
26
|
+
Accepts filtering via ScanConfig-compatible kwargs.
|
|
27
|
+
"""
|
|
28
|
+
# Normalize inputs into canonical form
|
|
29
|
+
kwargs = get_safe_canonical_kwargs(*args, **kwargs)
|
|
30
|
+
cfg = kwargs.get('cfg') or define_defaults(**kwargs)
|
|
31
|
+
|
|
32
|
+
# Get directory list (may come from args or kwargs)
|
|
33
|
+
kwargs["directories"] = ensure_directories(*args, **kwargs)
|
|
34
|
+
if not kwargs["directories"]:
|
|
35
|
+
return []
|
|
36
|
+
|
|
37
|
+
# Build base command for all directories
|
|
38
|
+
dir_expr = " ".join(shlex.quote(d) for d in kwargs["directories"])
|
|
39
|
+
cmd = [f"find {dir_expr}"]
|
|
40
|
+
|
|
41
|
+
# --- depth filters ---
|
|
42
|
+
if depth is not None:
|
|
43
|
+
cmd += [f"-mindepth {depth}", f"-maxdepth {depth}"]
|
|
44
|
+
else:
|
|
45
|
+
if mindepth is not None:
|
|
46
|
+
cmd.append(f"-mindepth {mindepth}")
|
|
47
|
+
if maxdepth is not None:
|
|
48
|
+
cmd.append(f"-maxdepth {maxdepth}")
|
|
49
|
+
|
|
50
|
+
# --- file type ---
|
|
51
|
+
if file_type in ("f", "d"):
|
|
52
|
+
cmd.append(f"-type {file_type}")
|
|
53
|
+
|
|
54
|
+
# --- basic attributes ---
|
|
55
|
+
if name:
|
|
56
|
+
cmd.append(f"-name {shlex.quote(name)}")
|
|
57
|
+
if size:
|
|
58
|
+
cmd.append(f"-size {shlex.quote(size)}")
|
|
59
|
+
if mtime:
|
|
60
|
+
cmd.append(f"-mtime {shlex.quote(mtime)}")
|
|
61
|
+
if perm:
|
|
62
|
+
cmd.append(f"-perm {shlex.quote(perm)}")
|
|
63
|
+
if user:
|
|
64
|
+
cmd.append(f"-user {shlex.quote(user)}")
|
|
65
|
+
|
|
66
|
+
# --- cfg-based filters ---
|
|
67
|
+
if cfg:
|
|
68
|
+
# Allowed extensions
|
|
69
|
+
if cfg.allowed_exts and cfg.allowed_exts != {"*"}:
|
|
70
|
+
ext_expr = " -o ".join(
|
|
71
|
+
[f"-name '*{e}'" for e in cfg.allowed_exts if e]
|
|
72
|
+
)
|
|
73
|
+
cmd.append(f"\\( {ext_expr} \\)")
|
|
74
|
+
|
|
75
|
+
# Excluded extensions
|
|
76
|
+
if cfg.exclude_exts:
|
|
77
|
+
for e in cfg.exclude_exts:
|
|
78
|
+
cmd.append(f"! -name '*{e}'")
|
|
79
|
+
|
|
80
|
+
# Allowed directories
|
|
81
|
+
if cfg.allowed_dirs and cfg.allowed_dirs != ["*"]:
|
|
82
|
+
dir_expr = " -o ".join(
|
|
83
|
+
[f"-path '*{d}*'" for d in cfg.allowed_dirs if d]
|
|
84
|
+
)
|
|
85
|
+
cmd.append(f"\\( {dir_expr} \\)")
|
|
86
|
+
|
|
87
|
+
# Excluded directories
|
|
88
|
+
if cfg.exclude_dirs:
|
|
89
|
+
for d in cfg.exclude_dirs:
|
|
90
|
+
cmd.append(f"! -path '*{d}*'")
|
|
91
|
+
|
|
92
|
+
# Allowed patterns
|
|
93
|
+
if cfg.allowed_patterns and cfg.allowed_patterns != ["*"]:
|
|
94
|
+
pat_expr = " -o ".join(
|
|
95
|
+
[f"-name '{p}'" for p in cfg.allowed_patterns if p]
|
|
96
|
+
)
|
|
97
|
+
cmd.append(f"\\( {pat_expr} \\)")
|
|
98
|
+
|
|
99
|
+
# Excluded patterns
|
|
100
|
+
if cfg.exclude_patterns:
|
|
101
|
+
for p in cfg.exclude_patterns:
|
|
102
|
+
cmd.append(f"! -name '{p}'")
|
|
103
|
+
|
|
104
|
+
# Allowed types (semantic, not `-type`)
|
|
105
|
+
if cfg.allowed_types and cfg.allowed_types != {"*"}:
|
|
106
|
+
type_expr = " -o ".join(
|
|
107
|
+
[f"-path '*{t}*'" for t in cfg.allowed_types if t]
|
|
108
|
+
)
|
|
109
|
+
cmd.append(f"\\( {type_expr} \\)")
|
|
110
|
+
|
|
111
|
+
# Excluded types
|
|
112
|
+
if cfg.exclude_types:
|
|
113
|
+
for t in cfg.exclude_types:
|
|
114
|
+
cmd.append(f"! -path '*{t}*'")
|
|
115
|
+
|
|
116
|
+
return " ".join(cmd)
|
|
117
|
+
|
|
118
|
+
|
|
119
|
+
|
|
120
|
+
def collect_globs(
|
|
121
|
+
*args,
|
|
122
|
+
mindepth: Optional[int] = None,
|
|
123
|
+
maxdepth: Optional[int] = None,
|
|
124
|
+
depth: Optional[int] = None,
|
|
125
|
+
file_type: Optional[str] = None, # "f", "d", or None
|
|
126
|
+
allowed: Optional[Callable[[str], bool]] = None,
|
|
127
|
+
**kwargs
|
|
128
|
+
) -> List[str] | dict:
|
|
129
|
+
"""
|
|
130
|
+
Collect file or directory paths recursively.
|
|
131
|
+
|
|
132
|
+
- If file_type is None → returns {"f": [...], "d": [...]}
|
|
133
|
+
- If file_type is "f" or "d" → returns a list of that type
|
|
134
|
+
- Supports SSH mode via `user_at_host`
|
|
135
|
+
"""
|
|
136
|
+
user_pass_host_key = get_user_pass_host_key(**kwargs)
|
|
137
|
+
kwargs["directories"] = ensure_directories(*args, **kwargs)
|
|
138
|
+
kwargs= get_safe_canonical_kwargs(**kwargs)
|
|
139
|
+
kwargs["cfg"] = kwargs.get('cfg') or define_defaults(**kwargs)
|
|
140
|
+
|
|
141
|
+
type_strs = {"f":"files","d":"dirs"}
|
|
142
|
+
file_type = get_proper_type_str(file_type)
|
|
143
|
+
file_types = make_list(file_type)
|
|
144
|
+
if file_type == None:
|
|
145
|
+
file_types = ["f","d"]
|
|
146
|
+
return_results = {}
|
|
147
|
+
return_result=[]
|
|
148
|
+
for file_type in file_types:
|
|
149
|
+
type_str = type_strs.get(file_type)
|
|
150
|
+
# Remote path (SSH)
|
|
151
|
+
find_cmd = get_find_cmd(
|
|
152
|
+
directories=kwargs.get("directories"),
|
|
153
|
+
cfg=kwargs.get('cfg'),
|
|
154
|
+
mindepth=mindepth,
|
|
155
|
+
maxdepth=maxdepth,
|
|
156
|
+
depth=depth,
|
|
157
|
+
file_type=file_type,
|
|
158
|
+
**user_pass_host_key,
|
|
159
|
+
)
|
|
160
|
+
result = run_pruned_func(run_cmd,find_cmd,
|
|
161
|
+
**kwargs
|
|
162
|
+
|
|
163
|
+
)
|
|
164
|
+
return_result = [res for res in result.split('\n') if res]
|
|
165
|
+
return_results[type_str]=return_result
|
|
166
|
+
if len(file_types) == 1:
|
|
167
|
+
return return_result
|
|
168
|
+
return return_results
|
|
169
|
+
def get_files_and_dirs(
|
|
170
|
+
*args,
|
|
171
|
+
recursive: bool = True,
|
|
172
|
+
include_files: bool = True,
|
|
173
|
+
**kwargs
|
|
174
|
+
):
|
|
175
|
+
if recursive == False:
|
|
176
|
+
kwargs['maxdepth']=1
|
|
177
|
+
if include_files == False:
|
|
178
|
+
kwargs['file_type']='d'
|
|
179
|
+
result = collect_globs(*args,**kwargs)
|
|
180
|
+
if include_files == False:
|
|
181
|
+
return result,[]
|
|
182
|
+
dirs = result.get("dirs")
|
|
183
|
+
files = result.get("files")
|
|
184
|
+
return dirs,files
|
|
185
|
+
def collect_filepaths(
|
|
186
|
+
*args,
|
|
187
|
+
**kwargs
|
|
188
|
+
) -> List[str]:
|
|
189
|
+
kwargs['file_type']='f'
|
|
190
|
+
return collect_globs(*args,**kwargs)
|
|
191
|
+
|
|
192
|
+
def get_filename(path):
|
|
193
|
+
basename = os.path.basename(path)
|
|
194
|
+
filename,ext = os.path.splitext(basename)
|
|
195
|
+
return filename
|
|
196
|
+
def find_files(filename,directory=None,add=None):
|
|
197
|
+
add = if_not_bool_default(add,default=True)
|
|
198
|
+
directory = directory or os.getcwd()
|
|
199
|
+
dirs,files = get_files_and_dirs(directory,add=add)
|
|
200
|
+
return [file for file in files if get_filename(file) == filename]
|
|
@@ -0,0 +1,210 @@
|
|
|
1
|
+
from .file_filters import *
|
|
2
|
+
from .reader_utils import *
|
|
3
|
+
from .find_collect import *
|
|
4
|
+
STOP_SEARCH = False
|
|
5
|
+
|
|
6
|
+
def request_find_console_stop():
|
|
7
|
+
global STOP_SEARCH
|
|
8
|
+
STOP_SEARCH = True
|
|
9
|
+
|
|
10
|
+
def reset_find_console_stop():
|
|
11
|
+
global STOP_SEARCH
|
|
12
|
+
STOP_SEARCH = False
|
|
13
|
+
|
|
14
|
+
def get_contents(
|
|
15
|
+
full_path=None,
|
|
16
|
+
parse_lines=False,
|
|
17
|
+
content=None
|
|
18
|
+
):
|
|
19
|
+
if full_path:
|
|
20
|
+
content = content or read_any_file(full_path)
|
|
21
|
+
if content:
|
|
22
|
+
if parse_lines:
|
|
23
|
+
content = str(content).split('\n')
|
|
24
|
+
return make_list(content)
|
|
25
|
+
return []
|
|
26
|
+
|
|
27
|
+
def _normalize(s: str, strip_comments=True, collapse_ws=True, lower=True):
|
|
28
|
+
if s is None:
|
|
29
|
+
return ""
|
|
30
|
+
if strip_comments:
|
|
31
|
+
s = s.split('//', 1)[0]
|
|
32
|
+
if collapse_ws:
|
|
33
|
+
s = re.sub(r'\s+', ' ', s)
|
|
34
|
+
if lower:
|
|
35
|
+
s = s.lower()
|
|
36
|
+
return s.strip()
|
|
37
|
+
|
|
38
|
+
def stringInContent(content, strings, total_strings=False, normalize=False):
|
|
39
|
+
if not content:
|
|
40
|
+
return False
|
|
41
|
+
if normalize:
|
|
42
|
+
c = _normalize(str(content))
|
|
43
|
+
|
|
44
|
+
found = [s for s in strings if _normalize(s) and _normalize(s) in c]
|
|
45
|
+
else:
|
|
46
|
+
c = str(content)
|
|
47
|
+
found = [s for s in strings if s and s in c]
|
|
48
|
+
if not found:
|
|
49
|
+
return False
|
|
50
|
+
return len(found) == len(strings) if total_strings else True
|
|
51
|
+
def find_file(content, spec_line, strings, total_strings=False):
|
|
52
|
+
lines = content.split('\n')
|
|
53
|
+
if 1 <= spec_line <= len(lines):
|
|
54
|
+
return stringInContent(lines[spec_line - 1], strings, total_strings=total_strings)
|
|
55
|
+
return False
|
|
56
|
+
def find_lines(content, strings, total_strings=False, normalize=True, any_per_line=True):
|
|
57
|
+
lines = content.split('\n')
|
|
58
|
+
hits = []
|
|
59
|
+
for i, line in enumerate(lines):
|
|
60
|
+
# match one line either if ANY string matches or if ALL match (configurable)
|
|
61
|
+
if any_per_line:
|
|
62
|
+
match = stringInContent(line, strings, total_strings=False, normalize=normalize)
|
|
63
|
+
else:
|
|
64
|
+
match = stringInContent(line, strings, total_strings=True, normalize=normalize)
|
|
65
|
+
if match:
|
|
66
|
+
hits.append({"line": i+1, "content": line})
|
|
67
|
+
return hits
|
|
68
|
+
def getPaths(files, strings):
|
|
69
|
+
tot_strings = strings
|
|
70
|
+
nu_files, found_paths = [], []
|
|
71
|
+
if isinstance(strings,list):
|
|
72
|
+
if len(strings) >1:
|
|
73
|
+
tot_strings = '\n'.join(strings)
|
|
74
|
+
else:
|
|
75
|
+
if len(strings) == 0:
|
|
76
|
+
return nu_files, found_paths
|
|
77
|
+
tot_strings = strings[0]
|
|
78
|
+
|
|
79
|
+
|
|
80
|
+
for file_path in files:
|
|
81
|
+
try:
|
|
82
|
+
og_content = read_any_file(file_path)
|
|
83
|
+
if tot_strings not in og_content:
|
|
84
|
+
continue
|
|
85
|
+
if file_path not in nu_files:
|
|
86
|
+
nu_files.append(file_path)
|
|
87
|
+
ogLines = og_content.split('\n')
|
|
88
|
+
# find all occurrences of the block
|
|
89
|
+
for m in re.finditer(re.escape(tot_strings), og_content):
|
|
90
|
+
start_line = og_content[:m.start()].count('\n') + 1 # 1-based
|
|
91
|
+
curr = {'file_path': file_path, 'lines': []}
|
|
92
|
+
for j in range(len(strings)):
|
|
93
|
+
ln = start_line + j
|
|
94
|
+
curr['lines'].append({'line': ln, 'content': ogLines[ln - 1]})
|
|
95
|
+
found_paths.append(curr)
|
|
96
|
+
except Exception as e:
|
|
97
|
+
print(f"{e}")
|
|
98
|
+
return nu_files, found_paths
|
|
99
|
+
|
|
100
|
+
def findContent(
|
|
101
|
+
*args,
|
|
102
|
+
strings: list=[],
|
|
103
|
+
total_strings=True,
|
|
104
|
+
parse_lines=False,
|
|
105
|
+
spec_line=False,
|
|
106
|
+
get_lines=True,
|
|
107
|
+
diffs=False,
|
|
108
|
+
**kwargs
|
|
109
|
+
):
|
|
110
|
+
global STOP_SEARCH
|
|
111
|
+
kwargs["directories"] = ensure_directories(*args,**kwargs)
|
|
112
|
+
|
|
113
|
+
found_paths = []
|
|
114
|
+
|
|
115
|
+
dirs, files = get_files_and_dirs(
|
|
116
|
+
**kwargs
|
|
117
|
+
)
|
|
118
|
+
nu_files, found_paths = getPaths(files, strings)
|
|
119
|
+
|
|
120
|
+
if diffs and found_paths:
|
|
121
|
+
return found_paths
|
|
122
|
+
|
|
123
|
+
for file_path in nu_files:
|
|
124
|
+
if STOP_SEARCH:
|
|
125
|
+
return found_paths # early exit
|
|
126
|
+
|
|
127
|
+
if file_path:
|
|
128
|
+
og_content = read_any_file(file_path)
|
|
129
|
+
contents = get_contents(
|
|
130
|
+
file_path,
|
|
131
|
+
parse_lines=parse_lines,
|
|
132
|
+
content=og_content
|
|
133
|
+
)
|
|
134
|
+
found = False
|
|
135
|
+
for content in contents:
|
|
136
|
+
if STOP_SEARCH:
|
|
137
|
+
return found_paths # bail out cleanly
|
|
138
|
+
|
|
139
|
+
if stringInContent(content, strings, total_strings=True, normalize=True):
|
|
140
|
+
found = True
|
|
141
|
+
if spec_line:
|
|
142
|
+
found = find_file(og_content, spec_line, strings, total_strings=True)
|
|
143
|
+
if found:
|
|
144
|
+
if get_lines:
|
|
145
|
+
lines = find_lines(
|
|
146
|
+
og_content,
|
|
147
|
+
strings=strings,
|
|
148
|
+
total_strings=False,
|
|
149
|
+
normalize=True,
|
|
150
|
+
any_per_line=True
|
|
151
|
+
)
|
|
152
|
+
if lines:
|
|
153
|
+
file_path = {"file_path": file_path, "lines": lines}
|
|
154
|
+
found_paths.append(file_path)
|
|
155
|
+
break
|
|
156
|
+
return found_paths
|
|
157
|
+
def return_function(start_dir=None,preferred_dir=None,basenames=None,functionName=None):
|
|
158
|
+
if basenames:
|
|
159
|
+
basenames = make_list(basenames)
|
|
160
|
+
abstract_file_finder = AbstractFileFinderImporter(start_dir=start_dir,preferred_dir=preferred_dir)
|
|
161
|
+
paths = abstract_file_finder.find_paths(basenames)
|
|
162
|
+
func = abstract_file_finder.import_function_from_path(paths[0], functionName)
|
|
163
|
+
return func
|
|
164
|
+
def getLineNums(file_path):
|
|
165
|
+
lines=[]
|
|
166
|
+
if file_path and isinstance(file_path,dict):
|
|
167
|
+
lines = file_path.get('lines')
|
|
168
|
+
file_path = file_path.get('file_path')
|
|
169
|
+
return file_path,lines
|
|
170
|
+
def get_line_content(obj):
|
|
171
|
+
line,content=None,None
|
|
172
|
+
if obj and isinstance(obj,dict):
|
|
173
|
+
line=obj.get('line')
|
|
174
|
+
content = obj.get('content')
|
|
175
|
+
#print(f"line: {line}\ncontent: {content}")
|
|
176
|
+
return line,content
|
|
177
|
+
def get_edit(file_path):
|
|
178
|
+
if file_path and os.path.isfile(file_path):
|
|
179
|
+
os.system(f"code {file_path}")
|
|
180
|
+
input()
|
|
181
|
+
def editLines(file_paths):
|
|
182
|
+
for file_path in file_paths:
|
|
183
|
+
file_path,lines = getLineNums(file_path)
|
|
184
|
+
for obj in lines:
|
|
185
|
+
line,content = get_line_content(obj)
|
|
186
|
+
get_edit(file_path)
|
|
187
|
+
def findContentAndEdit(*args,
|
|
188
|
+
strings: list=[],
|
|
189
|
+
total_strings=True,
|
|
190
|
+
parse_lines=False,
|
|
191
|
+
spec_line=False,
|
|
192
|
+
get_lines=True,
|
|
193
|
+
edit_lines=False,
|
|
194
|
+
diffs=False,
|
|
195
|
+
**kwargs
|
|
196
|
+
):
|
|
197
|
+
file_paths = findContent(
|
|
198
|
+
*args,
|
|
199
|
+
strings=strings,
|
|
200
|
+
total_strings=total_strings,
|
|
201
|
+
parse_lines=parse_lines,
|
|
202
|
+
spec_line=spec_line,
|
|
203
|
+
get_lines=get_lines,
|
|
204
|
+
diffs=diffs,
|
|
205
|
+
**kwargs
|
|
206
|
+
)
|
|
207
|
+
if edit_lines:
|
|
208
|
+
editLines(file_paths)
|
|
209
|
+
return file_paths
|
|
210
|
+
|