abstract-utilities 0.2.2.444__tar.gz → 0.2.2.490__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (101) hide show
  1. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/PKG-INFO +1 -1
  2. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/setup.py +1 -1
  3. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/__init__.py +2 -1
  4. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/class_utils.py +0 -1
  5. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/file_utils/file_utils/__init__.py +2 -0
  6. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/file_utils/file_utils/file_utils.py +3 -3
  7. abstract_utilities-0.2.2.490/src/abstract_utilities/file_utils/file_utils/find_collect.py +154 -0
  8. abstract_utilities-0.2.2.490/src/abstract_utilities/file_utils/file_utils/imports/__init__.py +3 -0
  9. abstract_utilities-0.2.2.490/src/abstract_utilities/file_utils/file_utils/imports/imports.py +39 -0
  10. abstract_utilities-0.2.2.490/src/abstract_utilities/file_utils/file_utils/imports/module_imports.py +14 -0
  11. abstract_utilities-0.2.2.490/src/abstract_utilities/file_utils/file_utils/type_checks.py +92 -0
  12. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/file_utils/imports/__init__.py +1 -2
  13. abstract_utilities-0.2.2.490/src/abstract_utilities/file_utils/imports/clean_imps.py +158 -0
  14. abstract_utilities-0.2.2.490/src/abstract_utilities/file_utils/imports/constants.py +39 -0
  15. abstract_utilities-0.2.2.490/src/abstract_utilities/file_utils/imports/imports.py +65 -0
  16. abstract_utilities-0.2.2.490/src/abstract_utilities/file_utils/imports/module_imports.py +13 -0
  17. abstract_utilities-0.2.2.490/src/abstract_utilities/read_write_utils.py +309 -0
  18. abstract_utilities-0.2.2.490/src/abstract_utilities/robust_reader/imports/imports.py +3 -0
  19. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_readers/import_utils/__init__.py +1 -0
  20. abstract_utilities-0.2.2.490/src/abstract_utilities/robust_readers/import_utils/clean_imports.py +175 -0
  21. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_readers/initFuncGen.py +10 -2
  22. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/ssh_utils/imports.py +1 -2
  23. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/string_clean.py +40 -1
  24. abstract_utilities-0.2.2.490/src/abstract_utilities/string_utils.py +51 -0
  25. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/type_utils.py +25 -1
  26. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities.egg-info/PKG-INFO +1 -1
  27. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities.egg-info/SOURCES.txt +9 -2
  28. abstract_utilities-0.2.2.444/src/abstract_utilities/file_utils/file_utils/imports.py +0 -1
  29. abstract_utilities-0.2.2.444/src/abstract_utilities/file_utils/imports/file_functions.py +0 -10
  30. abstract_utilities-0.2.2.444/src/abstract_utilities/file_utils/imports/imports.py +0 -14
  31. abstract_utilities-0.2.2.444/src/abstract_utilities/file_utils/imports/module_imports.py +0 -9
  32. abstract_utilities-0.2.2.444/src/abstract_utilities/read_write_utils.py +0 -142
  33. abstract_utilities-0.2.2.444/src/abstract_utilities/robust_reader/imports/imports.py +0 -12
  34. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/README.md +0 -0
  35. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/pyproject.toml +0 -0
  36. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/setup.cfg +0 -0
  37. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/abstract_classes.py +0 -0
  38. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/cmd_utils/__init__.py +0 -0
  39. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/cmd_utils/cmd_utils.py +0 -0
  40. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/cmd_utils/imports/__init__.py +0 -0
  41. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/cmd_utils/imports/imports.py +0 -0
  42. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/cmd_utils/pexpect_utils.py +0 -0
  43. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/cmd_utils/user_utils.py +0 -0
  44. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/collator_utils.py +0 -0
  45. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/compare_utils/__init__.py +0 -0
  46. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/compare_utils/best_match.py +0 -0
  47. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/compare_utils/compare_utils.py +0 -0
  48. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/compare_utils/find_value.py +0 -0
  49. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/doit.py +0 -0
  50. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/dynimport.py +0 -0
  51. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/env_utils/__init__.py +0 -0
  52. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/env_utils/abstractEnv.py +0 -0
  53. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/env_utils/envy_it.py +0 -0
  54. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/env_utils/imports/__init__.py +0 -0
  55. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/env_utils/imports/imports.py +0 -0
  56. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/env_utils/imports/utils.py +0 -0
  57. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/error_utils.py +0 -0
  58. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/file_utils/__init__.py +0 -0
  59. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/file_utils/file_utils/file_filters.py +0 -0
  60. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/file_utils/file_utils/file_reader.py +0 -0
  61. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/file_utils/file_utils/filter_params.py +0 -0
  62. {abstract_utilities-0.2.2.444/src/abstract_utilities → abstract_utilities-0.2.2.490/src/abstract_utilities/file_utils}/file_utils/imports/constants.py +0 -0
  63. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/file_utils/file_utils/map_utils.py +0 -0
  64. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/file_utils/file_utils/pdf_utils.py +0 -0
  65. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/file_utils/imports/classes.py +0 -0
  66. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/file_utils/req.py +0 -0
  67. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/global_utils.py +0 -0
  68. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/hash_utils.py +0 -0
  69. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/history_utils.py +0 -0
  70. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/json_utils.py +0 -0
  71. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/list_utils.py +0 -0
  72. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/log_utils.py +0 -0
  73. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/math_utils.py +0 -0
  74. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/parse_utils.py +0 -0
  75. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/path_utils.py +0 -0
  76. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_reader/__init__.py +0 -0
  77. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_reader/file_reader2.py +0 -0
  78. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_reader/file_readers.py +0 -0
  79. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_reader/imports/__init__.py +0 -0
  80. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_reader/sadfsad.py +0 -0
  81. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_readers/__init__.py +0 -0
  82. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_readers/import_utils/dot_utils.py +0 -0
  83. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_readers/import_utils/function_utils.py +0 -0
  84. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_readers/import_utils/import_utils.py +0 -0
  85. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_readers/import_utils/impot_functions.py +0 -0
  86. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_readers/import_utils/safe_import_utils.py +0 -0
  87. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_readers/import_utils/sysroot_utils.py +0 -0
  88. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_readers/import_utils/utils.py +0 -0
  89. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/robust_readers/imports.py +0 -0
  90. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/safe_utils.py +0 -0
  91. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/ssh_utils/__init__.py +0 -0
  92. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/ssh_utils/classes.py +0 -0
  93. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/ssh_utils/pexpect_utils.py +0 -0
  94. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/ssh_utils/utils.py +0 -0
  95. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/tetsts.py +0 -0
  96. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/thread_utils.py +0 -0
  97. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/time_utils.py +0 -0
  98. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities/utils.py +0 -0
  99. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities.egg-info/dependency_links.txt +0 -0
  100. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities.egg-info/requires.txt +0 -0
  101. {abstract_utilities-0.2.2.444 → abstract_utilities-0.2.2.490}/src/abstract_utilities.egg-info/top_level.txt +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: abstract_utilities
3
- Version: 0.2.2.444
3
+ Version: 0.2.2.490
4
4
  Summary: abstract_utilities is a collection of utility modules providing a variety of functions to aid in tasks such as data comparison, list manipulation, JSON handling, string manipulation, mathematical computations, and time operations.
5
5
  Home-page: https://github.com/AbstractEndeavors/abstract_utilities
6
6
  Author: putkoff
@@ -4,7 +4,7 @@ with open("README.md", "r", encoding="utf-8") as fh:
4
4
  long_description = fh.read()
5
5
  setuptools.setup(
6
6
  name='abstract_utilities',
7
- version='0.2.2.444',
7
+ version='0.2.2.490',
8
8
  author='putkoff',
9
9
  author_email='partners@abstractendeavors.com',
10
10
  description='abstract_utilities is a collection of utility modules providing a variety of functions to aid in tasks such as data comparison, list manipulation, JSON handling, string manipulation, mathematical computations, and time operations.',
@@ -116,9 +116,10 @@ from .parse_utils import (num_tokens_from_string,
116
116
 
117
117
  from .log_utils import get_caller_info,get_logFile,print_or_log,get_json_call_response,initialize_call_log
118
118
  from .error_utils import try_func
119
- from .class_utils import alias,get_class_inputs,get_set_attr
120
119
  from .ssh_utils import *
121
120
  from .env_utils import *
122
121
  from .path_utils import *
123
122
  from .file_utils import *
124
123
  from .file_utils import call_for_all_tabs
124
+ from .string_utils import *
125
+ from .class_utils import alias,get_class_inputs,get_set_attr,get_caller_path,get_caller_dir
@@ -405,4 +405,3 @@ def get_caller_dir(i: Optional[int] = None) -> str:
405
405
  abspath = get_caller_path(depth + 1)
406
406
  return os.path.dirname(abspath)
407
407
 
408
-
@@ -4,3 +4,5 @@ from .filter_params import *
4
4
  from .map_utils import *
5
5
  from .pdf_utils import *
6
6
  from .file_reader import *
7
+ from .find_collect import *
8
+ from .type_checks import *
@@ -1,7 +1,7 @@
1
- from pathlib import Path
2
- from typing import *
3
- import fnmatch, os, glob
1
+
4
2
  from .filter_params import *
3
+ from .imports import *
4
+
5
5
  ##from abstract_utilities import make_list,get_media_exts, is_media_type
6
6
  def get_allowed_predicate(allowed=None):
7
7
  if allowed != False:
@@ -0,0 +1,154 @@
1
+ from .imports import *
2
+ from .filter_params import *
3
+ from .file_filters import enumerate_source_files
4
+
5
+
6
+ def check_path_type(
7
+ path: str,
8
+ user: Optional[str] = None,
9
+ host: Optional[str] = None,
10
+ user_as_host: Optional[str] = None,
11
+ use_shell: bool = False
12
+ ) -> Literal["file", "directory", "missing", "unknown"]:
13
+ """
14
+ Determine whether a given path is a file, directory, or missing.
15
+ Works locally or remotely (via SSH).
16
+
17
+ Args:
18
+ path: The path to check.
19
+ user, host, user_as_host: SSH parameters if remote.
20
+ use_shell: Force shell test instead of Python os.path.
21
+ Returns:
22
+ One of: 'file', 'directory', 'missing', or 'unknown'
23
+ """
24
+
25
+ # --- remote check if user/host is given ---
26
+ if user_as_host or (user and host):
27
+ remote_target = user_as_host or f"{user}@{host}"
28
+ cmd = f"if [ -f '{path}' ]; then echo file; elif [ -d '{path}' ]; then echo directory; else echo missing; fi"
29
+ try:
30
+ result = subprocess.check_output(
31
+ ["ssh", remote_target, cmd],
32
+ stderr=subprocess.DEVNULL,
33
+ text=True,
34
+ timeout=5
35
+ ).strip()
36
+ return result if result in ("file", "directory", "missing") else "unknown"
37
+ except Exception:
38
+ return "unknown"
39
+
40
+ # --- local check ---
41
+ if not use_shell:
42
+ if os.path.isfile(path):
43
+ return "file"
44
+ elif os.path.isdir(path):
45
+ return "directory"
46
+ elif not os.path.exists(path):
47
+ return "missing"
48
+ return "unknown"
49
+ else:
50
+ # fallback using shell tests (useful for sandboxed contexts)
51
+ cmd = f"if [ -f '{path}' ]; then echo file; elif [ -d '{path}' ]; then echo directory; else echo missing; fi"
52
+ try:
53
+ output = subprocess.check_output(
54
+ cmd, shell=True, stderr=subprocess.DEVNULL, text=True
55
+ ).strip()
56
+ return output if output in ("file", "directory", "missing") else "unknown"
57
+ except Exception:
58
+ return "unknown"
59
+
60
+
61
+
62
+
63
+ def get_find_cmd(
64
+ directory: str,
65
+ *,
66
+ mindepth: Optional[int] = None,
67
+ maxdepth: Optional[int] = None,
68
+ depth: Optional[int] = None,
69
+ file_type: Optional[str] = None, # 'f' or 'd'
70
+ name: Optional[str] = None,
71
+ size: Optional[str] = None,
72
+ mtime: Optional[str] = None,
73
+ perm: Optional[str] = None,
74
+ user: Optional[str] = None,
75
+ **kwargs
76
+ ) -> str:
77
+ """Constructs a Unix `find` command string from keyword args."""
78
+ cmd = [f"find {directory}"]
79
+
80
+ if depth is not None:
81
+ cmd += [f"-mindepth {depth}", f"-maxdepth {depth}"]
82
+ else:
83
+ if mindepth is not None:
84
+ cmd.append(f"-mindepth {mindepth}")
85
+ if maxdepth is not None:
86
+ cmd.append(f"-maxdepth {maxdepth}")
87
+
88
+ if file_type in ("f", "d"):
89
+ cmd.append(f"-type {file_type}")
90
+ if name:
91
+ cmd.append(f"-name '{name}'")
92
+ if size:
93
+ cmd.append(f"-size {size}")
94
+ if mtime:
95
+ cmd.append(f"-mtime {mtime}")
96
+ if perm:
97
+ cmd.append(f"-perm {perm}")
98
+ if user:
99
+ cmd.append(f"-user {user}")
100
+
101
+ return " ".join(cmd)
102
+
103
+
104
+ def collect_globs(
105
+ directory: str,
106
+ cfg: Optional["ScanConfig"] = None,
107
+ *,
108
+ exts: Optional[Set[str]] = None,
109
+ patterns: Optional[List[str]] = None,
110
+ mindepth: Optional[int] = None,
111
+ maxdepth: Optional[int] = None,
112
+ depth: Optional[int] = None,
113
+ file_type: Optional[str] = None,
114
+ user_at_host: Optional[str] = None,
115
+ add: bool = False,
116
+ **kwargs
117
+ ) -> List[str]:
118
+ """
119
+ Collect file or directory paths using either:
120
+ - local recursive logic (rglob)
121
+ - or remote shell call (find via run_cmd)
122
+ """
123
+ cfg = cfg or define_defaults(add=add)
124
+ directory = str(directory)
125
+ exts = ensure_exts(exts)
126
+ patterns = ensure_patterns(patterns)
127
+
128
+ # Remote path via SSH
129
+ if user_at_host:
130
+ find_cmd = get_find_cmd(
131
+ directory,
132
+ mindepth=mindepth,
133
+ maxdepth=maxdepth,
134
+ depth=depth,
135
+ file_type=file_type,
136
+ **{k: v for k, v in kwargs.items() if v},
137
+ )
138
+ return run_cmd(find_cmd, user_at_host=user_at_host)
139
+
140
+ # Local path (Python-native walk)
141
+ root = Path(directory)
142
+ results = []
143
+ for p in root.rglob("*"):
144
+ if file_type == "f" and not p.is_file():
145
+ continue
146
+ if file_type == "d" and not p.is_dir():
147
+ continue
148
+ if exts and p.suffix.lower() not in exts:
149
+ continue
150
+ if patterns and not any(p.match(pat) for pat in patterns):
151
+ continue
152
+ results.append(str(p.resolve()))
153
+
154
+ return sorted(results)
@@ -0,0 +1,3 @@
1
+ from .constants import *
2
+ from .imports import *
3
+ from ..imports import *
@@ -0,0 +1,39 @@
1
+ # ============================================================
2
+ # abstract_utilities/imports/imports.py
3
+ # Global imports hub — everything imported here will be
4
+ # automatically available to any module that does:
5
+ # from ..imports import *
6
+ # ============================================================
7
+ # ---- Core standard library modules -------------------------
8
+ import os, sys, re, shlex, glob, platform, textwrap, subprocess, inspect, json, time
9
+ import tempfile, shutil, logging, pathlib, fnmatch, importlib, importlib.util, types
10
+ from pathlib import Path
11
+ from datetime import datetime
12
+ from types import ModuleType
13
+
14
+ # ---- Dataclasses and typing --------------------------------
15
+ from dataclasses import dataclass, field
16
+ from typing import (
17
+ Any, Optional, List, Dict, Set, Tuple,
18
+ Iterable, Callable, Literal, Union, TypeVar
19
+ )
20
+
21
+ # ---- Common 3rd-party dependencies --------------------------
22
+ import pandas as pd
23
+ import geopandas as gpd
24
+ import pytesseract
25
+ import pdfplumber
26
+ import PyPDF2
27
+ import ezodf
28
+ from pdf2image import convert_from_path
29
+ from werkzeug.utils import secure_filename
30
+ from werkzeug.datastructures import FileStorage
31
+
32
+ # ---- Helpers ------------------------------------------------
33
+ import textwrap as tw
34
+ from pprint import pprint
35
+
36
+ # ============================================================
37
+ # AUTO-EXPORT ALL NON-PRIVATE NAMES
38
+ # ============================================================
39
+ __all__ = [name for name in globals() if not name.startswith("_")]
@@ -0,0 +1,14 @@
1
+ from .imports import *
2
+ from ....string_clean import eatAll
3
+ from ....list_utils import make_list
4
+ from ....type_utils import get_media_exts, is_media_type, MIME_TYPES, is_str
5
+ from ....ssh_utils import *
6
+ from ....env_utils import *
7
+ from ....read_write_utils import *
8
+ from ....abstract_classes import SingletonMeta
9
+ from ....string_utils import get_from_kwargs
10
+ from ....abstract_classes import run_pruned_func
11
+ from ....class_utils import get_caller, get_caller_path, get_caller_dir
12
+
13
+
14
+ __all__ = [name for name in globals() if not name.startswith("_")]
@@ -0,0 +1,92 @@
1
+ from .imports import *
2
+
3
+ def get_user_pass_host_key(**kwargs):
4
+ args = ['password','user_at_host','host','key','user']
5
+ kwargs['del_kwarg']=kwargs.get('del_kwarg',False)
6
+ values,kwargs = get_from_kwargs(*args,**kwargs)
7
+ return values
8
+
9
+ # --- Base remote checker -----------------------------------------------------
10
+ def _remote_test(path: str, test_flag: str, timeout: int = 5,*args, **kwargs) -> bool:
11
+ """
12
+ Run a remote shell test (e.g. -f, -d) via SSH.
13
+ Returns True if test succeeds, False otherwise.
14
+ """
15
+ try:
16
+ kwargs['cmd']=f"[ {test_flag} {shlex.quote(path)} ] && echo 1 || echo 0"
17
+ kwargs['text']=True
18
+ kwargs['timeout']=timeout
19
+ kwargs['stderr']=subprocess.DEVNULL
20
+ result = run_pruned_func(run_cmd,**kwargs)
21
+ return result.strip() == "1"
22
+ except Exception:
23
+ return False
24
+
25
+
26
+ # --- Individual path checks --------------------------------------------------
27
+ def is_remote_file(path: str,*args, **kwargs) -> bool:
28
+ """True if remote path is a file."""
29
+ return _remote_test(path, "-f", **kwargs)
30
+
31
+
32
+ def is_remote_dir(path: str,*args, **kwargs) -> bool:
33
+ """True if remote path is a directory."""
34
+ return _remote_test(path, "-d", **kwargs)
35
+
36
+
37
+ def is_local_file(path: str) -> bool:
38
+ """True if local path is a file."""
39
+ return os.path.isfile(path)
40
+
41
+
42
+ def is_local_dir(path: str) -> bool:
43
+ """True if local path is a directory."""
44
+ return os.path.isdir(path)
45
+
46
+
47
+ # --- Unified interface -------------------------------------------------------
48
+
49
+ def is_file(path: str,*args,**kwargs) -> bool:
50
+ """Determine if path is a file (works local or remote)."""
51
+ if get_user_pass_host_key(**kwargs):
52
+ return is_remote_file(path, **kwargs)
53
+ return is_local_file(path)
54
+
55
+
56
+ def is_dir(path: str, *args,**kwargs) -> bool:
57
+ """Determine if path is a directory (works local or remote)."""
58
+ if get_user_pass_host_key(**kwargs):
59
+ return is_remote_dir(path, **kwargs)
60
+ return is_local_dir(path)
61
+
62
+ def is_exists(path: str, *args,**kwargs) -> bool:
63
+ if is_file(path,**kwargs):
64
+ return True
65
+ if is_dir(path,**kwargs):
66
+ return True
67
+ return False
68
+ # --- Optional: keep your original all-in-one wrapper ------------------------
69
+ def check_path_type(
70
+ path: str,
71
+ *args,
72
+ **kwargs
73
+ ) -> str:
74
+ """
75
+ Return 'file', 'directory', 'missing', or 'unknown'.
76
+ Uses isolated is_file/is_dir functions.
77
+ """
78
+ if get_user_pass_host_key(**kwargs):
79
+ if is_remote_file(path,**kwargs):
80
+ return "file"
81
+ elif is_remote_dir(path,**kwargs):
82
+ return "directory"
83
+ else:
84
+ return "missing"
85
+ else:
86
+ if os.path.isfile(path):
87
+ return "file"
88
+ elif os.path.isdir(path):
89
+ return "directory"
90
+ elif not os.path.exists(path):
91
+ return "missing"
92
+ return "unknown"
@@ -1,5 +1,4 @@
1
1
  from .constants import *
2
- from .imports import *
3
2
  from .module_imports import *
4
3
  from .classes import *
5
- from .file_functions import *
4
+ from .imports import *
@@ -0,0 +1,158 @@
1
+ from abstract_utilities import read_from_file, eatAll
2
+ import os, sys, re, inspect
3
+ from typing import *
4
+
5
+ # ============================================================
6
+ # Constants
7
+ # ============================================================
8
+ import_tag = 'import '
9
+ from_tag = 'from '
10
+
11
+ # ============================================================
12
+ # Helpers
13
+ # ============================================================
14
+ def get_caller_path(i=None):
15
+ i = i or 1
16
+ frame = inspect.stack()[i]
17
+ return os.path.abspath(frame.filename)
18
+
19
+ def make_list(obj: any) -> list:
20
+ if isinstance(obj, str) and ',' in obj:
21
+ obj = obj.split(',')
22
+ if isinstance(obj, (set, tuple)):
23
+ return list(obj)
24
+ if isinstance(obj, list):
25
+ return obj
26
+ return [obj]
27
+
28
+ def eatElse(stringObj, chars=None):
29
+ chars = make_list(chars or []) + list('abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_')
30
+ while stringObj:
31
+ if stringObj and stringObj[0] not in chars:
32
+ stringObj = stringObj[1:]
33
+ continue
34
+ if stringObj and stringObj[-1] not in chars:
35
+ stringObj = stringObj[:-1]
36
+ continue
37
+ break
38
+ return stringObj
39
+
40
+ def clean_line(line):
41
+ return eatAll(line, [' ', '', '\t', '\n'])
42
+
43
+ def is_line_import(line):
44
+ return bool(line and line.startswith(import_tag) and 'from ' not in line)
45
+
46
+ def is_line_from_import(line):
47
+ return bool(line and line.startswith(from_tag) and ' import ' in line)
48
+
49
+ def is_from_group_start(line):
50
+ return bool(line and line.startswith(from_tag) and 'import' in line and '(' in line and not line.rstrip().endswith(')'))
51
+
52
+ def is_from_group_end(line):
53
+ return bool(line and ')' in line)
54
+
55
+ def clean_imports(imports):
56
+ if isinstance(imports, str):
57
+ imports = imports.split(',')
58
+ return [eatElse(imp.strip()) for imp in imports if imp.strip()]
59
+
60
+ # ============================================================
61
+ # Combine lone import statements
62
+ # ============================================================
63
+ def combine_lone_imports(text=None, file_path=None):
64
+ text = text or ''
65
+ if file_path and os.path.isfile(file_path):
66
+ text += read_from_file(file_path)
67
+ lines = text.split('\n')
68
+
69
+ cleaned_import_list = []
70
+ nu_lines = []
71
+ j = None
72
+
73
+ for i, line in enumerate(lines):
74
+ if is_line_import(line):
75
+ if j is None:
76
+ nu_lines.append(import_tag)
77
+ j = i
78
+ cleaned_import_list += clean_imports(line.split(import_tag)[1])
79
+ else:
80
+ nu_lines.append(line)
81
+
82
+ if j is None:
83
+ return '\n'.join(nu_lines)
84
+ cleaned_import_list = sorted(set(cleaned_import_list))
85
+ nu_lines[j] += ', '.join(cleaned_import_list)
86
+ return '\n'.join(nu_lines)
87
+
88
+ # ============================================================
89
+ # Merge repeated 'from pkg import ...' (1-line only)
90
+ # Preserve multi-line grouped imports
91
+ # ============================================================
92
+ def merge_from_import_groups(text=None, file_path=None):
93
+ if file_path and os.path.isfile(file_path):
94
+ text = read_from_file(file_path)
95
+ text = text or ''
96
+ lines = text.split('\n')
97
+
98
+ pkg_to_imports: Dict[str, Set[str]] = {}
99
+ pkg_to_line_index: Dict[str, int] = {}
100
+ nu_lines: List[str] = []
101
+
102
+ in_group = False
103
+ for i, line in enumerate(lines):
104
+ stripped = line.strip()
105
+
106
+ # preserve multi-line grouped blocks intact
107
+ if in_group:
108
+ nu_lines.append(line)
109
+ if is_from_group_end(line):
110
+ in_group = False
111
+ continue
112
+
113
+ if is_from_group_start(line):
114
+ in_group = True
115
+ nu_lines.append(line)
116
+ continue
117
+
118
+ if is_line_from_import(line):
119
+ try:
120
+ pkg_part, imps_part = line.split(' import ', 1)
121
+ pkg_name = pkg_part.replace('from ', '').strip()
122
+ imps = clean_imports(imps_part)
123
+ except Exception:
124
+ nu_lines.append(line)
125
+ continue
126
+
127
+ if pkg_name not in pkg_to_imports:
128
+ pkg_to_imports[pkg_name] = set(imps)
129
+ pkg_to_line_index[pkg_name] = len(nu_lines)
130
+ nu_lines.append(line)
131
+ else:
132
+ pkg_to_imports[pkg_name].update(imps)
133
+ else:
134
+ nu_lines.append(line)
135
+
136
+ # Rewrite first occurrences
137
+ for pkg, idx in pkg_to_line_index.items():
138
+ all_imps = sorted(pkg_to_imports[pkg])
139
+ nu_lines[idx] = f"from {pkg} import {', '.join(all_imps)}"
140
+
141
+ return '\n'.join(nu_lines)
142
+
143
+ # ============================================================
144
+ # Pipeline
145
+ # ============================================================
146
+ def clean_imports_pipeline(path: str):
147
+ raw = read_from_file(path)
148
+ step1 = combine_lone_imports(text=raw)
149
+ step2 = merge_from_import_groups(text=step1)
150
+ return step2
151
+
152
+ # ============================================================
153
+ # Standalone Run
154
+ # ============================================================
155
+ if __name__ == "__main__":
156
+ abs_path = "/home/flerb/Documents/pythonTools/modules/src/modules/abstract_utilities/src/abstract_utilities/file_utils/imports/imports.py"
157
+ cleaned = clean_imports_pipeline(abs_path)
158
+ print(cleaned)
@@ -0,0 +1,39 @@
1
+ from .imports import *
2
+ from .module_imports import *
3
+ @dataclass
4
+ class ScanConfig:
5
+ allowed_exts: Set[str]
6
+ unallowed_exts: Set[str]
7
+ exclude_types: Set[str]
8
+ exclude_dirs: List[str] = field(default_factory=list)
9
+ exclude_patterns: List[str] = field(default_factory=list)
10
+ DEFAULT_ALLOWED_EXTS: Set[str] = {
11
+ ".py", ".pyw", # python
12
+ ".js", ".jsx", ".ts", ".tsx", ".mjs", # JS/TS
13
+ ".html", ".htm", ".xml", # markup
14
+ ".css", ".scss", ".sass", ".less", # styles
15
+ ".json", ".yaml", ".yml", ".toml", ".ini", # configs
16
+ ".cfg", ".md", ".markdown", ".rst", # docs
17
+ ".sh", ".bash", ".env", # scripts/env
18
+ ".txt" # plain text
19
+ }
20
+
21
+ DEFAULT_EXCLUDE_TYPES: Set[str] = {
22
+ "image", "video", "audio", "presentation",
23
+ "spreadsheet", "archive", "executable"
24
+ }
25
+
26
+ # never want these—even if they sneak into ALLOWED
27
+ _unallowed = set(get_media_exts(DEFAULT_EXCLUDE_TYPES)) | {'.bak', '.shp', '.cpg', '.dbf', '.shx','.geojson',".pyc",'.shx','.geojson','.prj','.sbn','.sbx'}
28
+ DEFAULT_UNALLOWED_EXTS = {e for e in _unallowed if e not in DEFAULT_ALLOWED_EXTS}
29
+
30
+ DEFAULT_EXCLUDE_DIRS: Set[str] = {
31
+ "node_modules", "old","__pycache__", "backups", "backup", "backs", "trash", "depriciated", "old", "__init__"
32
+ }
33
+
34
+ DEFAULT_EXCLUDE_PATTERNS: Set[str] = {
35
+ "__init__*", "*.tmp", "*.log", "*.lock", "*.zip","*~"
36
+ }
37
+ REMOTE_RE = re.compile(r"^(?P<host>[^:\s]+@[^:\s]+):(?P<path>/.*)$")
38
+ AllowedPredicate = Optional[Callable[[str], bool]]
39
+ DEFAULT_EXCLUDE_FILE_PATTERNS=DEFAULT_EXCLUDE_PATTERNS
@@ -0,0 +1,65 @@
1
+ # ============================================================
2
+ # abstract_utilities/imports/imports.py
3
+ # Global imports hub — everything imported here will be
4
+ # automatically available to any module that does:
5
+ # from ..imports import *
6
+ # ============================================================
7
+
8
+
9
+ import os
10
+ import sys, importlib,os
11
+ import sys, importlib, os, inspect
12
+ from pathlib import Path
13
+ import os,sys
14
+
15
+
16
+
17
+ from typing import *
18
+ import re
19
+
20
+ from typing import *
21
+ from types import MethodType
22
+ import os,re, sys, importlib, inspect, os, importlib.util, hashlib
23
+ import os,tempfile,shutil,logging,ezodf,fnmatch,pytesseract,pdfplumber
24
+ import pandas as pd
25
+ import geopandas as gpd
26
+ from datetime import datetime
27
+
28
+ from typing import *
29
+ from werkzeug.utils import secure_filename
30
+ from werkzeug.datastructures import FileStorage
31
+ from pdf2image import convert_from_path # only used for OCR fallback
32
+ # ---- Core standard library modules -------------------------
33
+ import os, sys, re, shlex, glob, platform, textwrap, subprocess, inspect, json, time
34
+ import tempfile, shutil, logging, pathlib, fnmatch, importlib, importlib.util, types
35
+
36
+ from datetime import datetime
37
+ from types import ModuleType
38
+
39
+ # ---- Dataclasses and typing --------------------------------
40
+ from dataclasses import dataclass, field
41
+ from typing import (
42
+ Any, Optional, List, Dict, Set, Tuple,
43
+ Iterable, Callable, Literal, Union, TypeVar
44
+ )
45
+
46
+ # ---- Common 3rd-party dependencies --------------------------
47
+ import pandas as pd
48
+ import geopandas as gpd
49
+ import pytesseract
50
+ import pdfplumber
51
+ import PyPDF2
52
+ import ezodf
53
+ from pdf2image import convert_from_path
54
+ from werkzeug.utils import secure_filename
55
+ from werkzeug.datastructures import FileStorage
56
+
57
+ # ---- Helpers ------------------------------------------------
58
+ import textwrap as tw
59
+ from pprint import pprint
60
+
61
+ # ============================================================
62
+ # AUTO-EXPORT ALL NON-PRIVATE NAMES
63
+ # ============================================================
64
+ __all__ = [name for name in globals() if not name.startswith("_")]
65
+
@@ -0,0 +1,13 @@
1
+ from .imports import *
2
+ from ...string_clean import eatAll
3
+ from ...list_utils import make_list
4
+ from ...type_utils import get_media_exts, is_media_type, MIME_TYPES, is_str
5
+ from ...ssh_utils import *
6
+ from ...env_utils import *
7
+ from ...read_write_utils import *
8
+ from ...abstract_classes import SingletonMeta
9
+ from ...log_utils import get_logFile
10
+ from ...class_utils import get_caller, get_caller_path, get_caller_dir
11
+ from ...ssh_utils import run_cmd
12
+
13
+ __all__ = [name for name in globals() if not name.startswith("_")]