Compare commits
2 Commits
cc59e8b8da
...
5c38228a1c
| Author | SHA1 | Date | |
|---|---|---|---|
| 5c38228a1c | |||
| 9ce1dbadc1 |
BIN
__pycache__/main.cpython-312.pyc
Normal file
BIN
__pycache__/main.cpython-312.pyc
Normal file
Binary file not shown.
Binary file not shown.
2015
build/create_a_link/Analysis-00.toc
Normal file
2015
build/create_a_link/Analysis-00.toc
Normal file
File diff suppressed because it is too large
Load Diff
293
build/create_a_link/EXE-00.toc
Normal file
293
build/create_a_link/EXE-00.toc
Normal file
@@ -0,0 +1,293 @@
|
|||||||
|
('D:\\Projects\\cjgc_upload\\dist\\create_a_link.exe',
|
||||||
|
True,
|
||||||
|
False,
|
||||||
|
False,
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\PyInstaller\\bootloader\\images\\icon-console.ico',
|
||||||
|
None,
|
||||||
|
False,
|
||||||
|
False,
|
||||||
|
b'<?xml version="1.0" encoding="UTF-8" standalone="yes"?>\n<assembly xmlns='
|
||||||
|
b'"urn:schemas-microsoft-com:asm.v1" manifestVersion="1.0">\n <trustInfo x'
|
||||||
|
b'mlns="urn:schemas-microsoft-com:asm.v3">\n <security>\n <requested'
|
||||||
|
b'Privileges>\n <requestedExecutionLevel level="asInvoker" uiAccess='
|
||||||
|
b'"false"/>\n </requestedPrivileges>\n </security>\n </trustInfo>\n '
|
||||||
|
b'<compatibility xmlns="urn:schemas-microsoft-com:compatibility.v1">\n <'
|
||||||
|
b'application>\n <supportedOS Id="{e2011457-1546-43c5-a5fe-008deee3d3f'
|
||||||
|
b'0}"/>\n <supportedOS Id="{35138b9a-5d96-4fbd-8e2d-a2440225f93a}"/>\n '
|
||||||
|
b' <supportedOS Id="{4a2f28e3-53b9-4441-ba9c-d69d4a4a6e38}"/>\n <s'
|
||||||
|
b'upportedOS Id="{1f676c76-80e1-4239-95bb-83d0f6d0da78}"/>\n <supporte'
|
||||||
|
b'dOS Id="{8e0f7a12-bfb3-4fe8-b9a5-48fd50a15a9a}"/>\n </application>\n <'
|
||||||
|
b'/compatibility>\n <application xmlns="urn:schemas-microsoft-com:asm.v3">'
|
||||||
|
b'\n <windowsSettings>\n <longPathAware xmlns="http://schemas.micros'
|
||||||
|
b'oft.com/SMI/2016/WindowsSettings">true</longPathAware>\n </windowsSett'
|
||||||
|
b'ings>\n </application>\n <dependency>\n <dependentAssembly>\n <ass'
|
||||||
|
b'emblyIdentity type="win32" name="Microsoft.Windows.Common-Controls" version='
|
||||||
|
b'"6.0.0.0" processorArchitecture="*" publicKeyToken="6595b64144ccf1df" langua'
|
||||||
|
b'ge="*"/>\n </dependentAssembly>\n </dependency>\n</assembly>',
|
||||||
|
True,
|
||||||
|
False,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\create_a_link.pkg',
|
||||||
|
[('pyi-contents-directory _internal', '', 'OPTION'),
|
||||||
|
('PYZ-00.pyz',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\PYZ-00.pyz',
|
||||||
|
'PYZ'),
|
||||||
|
('struct',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\localpycs\\struct.pyc',
|
||||||
|
'PYMODULE'),
|
||||||
|
('pyimod01_archive',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\localpycs\\pyimod01_archive.pyc',
|
||||||
|
'PYMODULE'),
|
||||||
|
('pyimod02_importers',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\localpycs\\pyimod02_importers.pyc',
|
||||||
|
'PYMODULE'),
|
||||||
|
('pyimod03_ctypes',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\localpycs\\pyimod03_ctypes.pyc',
|
||||||
|
'PYMODULE'),
|
||||||
|
('pyimod04_pywin32',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\localpycs\\pyimod04_pywin32.pyc',
|
||||||
|
'PYMODULE'),
|
||||||
|
('pyiboot01_bootstrap',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\PyInstaller\\loader\\pyiboot01_bootstrap.py',
|
||||||
|
'PYSOURCE'),
|
||||||
|
('pyi_rth_inspect',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\PyInstaller\\hooks\\rthooks\\pyi_rth_inspect.py',
|
||||||
|
'PYSOURCE'),
|
||||||
|
('pyi_rth_pkgutil',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\PyInstaller\\hooks\\rthooks\\pyi_rth_pkgutil.py',
|
||||||
|
'PYSOURCE'),
|
||||||
|
('pyi_rth_multiprocessing',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\PyInstaller\\hooks\\rthooks\\pyi_rth_multiprocessing.py',
|
||||||
|
'PYSOURCE'),
|
||||||
|
('create_a_link', 'D:\\Projects\\cjgc_upload\\create_a_link.py', 'PYSOURCE'),
|
||||||
|
('selenium\\webdriver\\common\\windows\\selenium-manager.exe',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\windows\\selenium-manager.exe',
|
||||||
|
'BINARY'),
|
||||||
|
('python312.dll', 'C:\\Program Files\\Python312\\python312.dll', 'BINARY'),
|
||||||
|
('_multiprocessing.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_multiprocessing.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('pyexpat.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\pyexpat.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('_ssl.pyd', 'C:\\Program Files\\Python312\\DLLs\\_ssl.pyd', 'EXTENSION'),
|
||||||
|
('_hashlib.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_hashlib.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('unicodedata.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\unicodedata.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('_decimal.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_decimal.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('_socket.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_socket.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('select.pyd', 'C:\\Program Files\\Python312\\DLLs\\select.pyd', 'EXTENSION'),
|
||||||
|
('_lzma.pyd', 'C:\\Program Files\\Python312\\DLLs\\_lzma.pyd', 'EXTENSION'),
|
||||||
|
('_bz2.pyd', 'C:\\Program Files\\Python312\\DLLs\\_bz2.pyd', 'EXTENSION'),
|
||||||
|
('_ctypes.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_ctypes.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('_queue.pyd', 'C:\\Program Files\\Python312\\DLLs\\_queue.pyd', 'EXTENSION'),
|
||||||
|
('_overlapped.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_overlapped.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('_asyncio.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_asyncio.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('_wmi.pyd', 'C:\\Program Files\\Python312\\DLLs\\_wmi.pyd', 'EXTENSION'),
|
||||||
|
('_uuid.pyd', 'C:\\Program Files\\Python312\\DLLs\\_uuid.pyd', 'EXTENSION'),
|
||||||
|
('charset_normalizer\\md__mypyc.cp312-win_amd64.pyd',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\charset_normalizer\\md__mypyc.cp312-win_amd64.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('charset_normalizer\\md.cp312-win_amd64.pyd',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\charset_normalizer\\md.cp312-win_amd64.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('api-ms-win-core-synch-l1-2-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-synch-l1-2-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-conio-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-conio-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-environment-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-environment-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-convert-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-convert-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-time-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-time-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-string-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-string-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-filesystem-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-filesystem-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-locale-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-locale-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-process-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-process-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-math-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-math-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-heap-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-heap-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-runtime-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-runtime-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('VCRUNTIME140.dll',
|
||||||
|
'C:\\Program Files\\Python312\\VCRUNTIME140.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-stdio-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-stdio-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-utility-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-utility-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('libcrypto-3.dll',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\libcrypto-3.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('libssl-3.dll',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\libssl-3.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('libffi-8.dll',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\libffi-8.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('ucrtbase.dll', 'D:\\Resource\\JDK\\JavaJDK17\\bin\\ucrtbase.dll', 'BINARY'),
|
||||||
|
('api-ms-win-core-synch-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-synch-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-sysinfo-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-sysinfo-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-processthreads-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-processthreads-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-string-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-string-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-timezone-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-timezone-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-console-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-console-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-heap-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-heap-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-memory-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-memory-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-processenvironment-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-processenvironment-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-errorhandling-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-errorhandling-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-datetime-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-datetime-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-file-l1-2-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-file-l1-2-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-processthreads-l1-1-1.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-processthreads-l1-1-1.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-rtlsupport-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-rtlsupport-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-interlocked-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-interlocked-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-namedpipe-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-namedpipe-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-file-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-file-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-handle-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-handle-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-profile-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-profile-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-file-l2-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-file-l2-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-libraryloader-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-libraryloader-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-util-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-util-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-localization-l1-2-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-localization-l1-2-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-debug-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-debug-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('selenium\\webdriver\\common\\devtools\\v139\\py.typed',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\devtools\\v139\\py.typed',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\remote\\isDisplayed.js',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\remote\\isDisplayed.js',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\common\\linux\\selenium-manager',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\linux\\selenium-manager',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\common\\macos\\selenium-manager',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\macos\\selenium-manager',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\firefox\\webdriver_prefs.json',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\firefox\\webdriver_prefs.json',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\common\\devtools\\v138\\py.typed',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\devtools\\v138\\py.typed',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\common\\devtools\\v137\\py.typed',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\devtools\\v137\\py.typed',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\remote\\findElements.js',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\remote\\findElements.js',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\common\\mutation-listener.js',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\mutation-listener.js',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\remote\\getAttribute.js',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\remote\\getAttribute.js',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\py.typed',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\py.typed',
|
||||||
|
'DATA'),
|
||||||
|
('certifi\\cacert.pem',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\certifi\\cacert.pem',
|
||||||
|
'DATA'),
|
||||||
|
('certifi\\py.typed',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\certifi\\py.typed',
|
||||||
|
'DATA'),
|
||||||
|
('base_library.zip',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\base_library.zip',
|
||||||
|
'DATA')],
|
||||||
|
[],
|
||||||
|
False,
|
||||||
|
False,
|
||||||
|
1770274794,
|
||||||
|
[('run.exe',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\PyInstaller\\bootloader\\Windows-64bit-intel\\run.exe',
|
||||||
|
'EXECUTABLE')],
|
||||||
|
'C:\\Program Files\\Python312\\python312.dll')
|
||||||
269
build/create_a_link/PKG-00.toc
Normal file
269
build/create_a_link/PKG-00.toc
Normal file
@@ -0,0 +1,269 @@
|
|||||||
|
('D:\\Projects\\cjgc_upload\\build\\create_a_link\\create_a_link.pkg',
|
||||||
|
{'BINARY': True,
|
||||||
|
'DATA': True,
|
||||||
|
'EXECUTABLE': True,
|
||||||
|
'EXTENSION': True,
|
||||||
|
'PYMODULE': True,
|
||||||
|
'PYSOURCE': True,
|
||||||
|
'PYZ': False,
|
||||||
|
'SPLASH': True,
|
||||||
|
'SYMLINK': False},
|
||||||
|
[('pyi-contents-directory _internal', '', 'OPTION'),
|
||||||
|
('PYZ-00.pyz',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\PYZ-00.pyz',
|
||||||
|
'PYZ'),
|
||||||
|
('struct',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\localpycs\\struct.pyc',
|
||||||
|
'PYMODULE'),
|
||||||
|
('pyimod01_archive',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\localpycs\\pyimod01_archive.pyc',
|
||||||
|
'PYMODULE'),
|
||||||
|
('pyimod02_importers',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\localpycs\\pyimod02_importers.pyc',
|
||||||
|
'PYMODULE'),
|
||||||
|
('pyimod03_ctypes',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\localpycs\\pyimod03_ctypes.pyc',
|
||||||
|
'PYMODULE'),
|
||||||
|
('pyimod04_pywin32',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\localpycs\\pyimod04_pywin32.pyc',
|
||||||
|
'PYMODULE'),
|
||||||
|
('pyiboot01_bootstrap',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\PyInstaller\\loader\\pyiboot01_bootstrap.py',
|
||||||
|
'PYSOURCE'),
|
||||||
|
('pyi_rth_inspect',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\PyInstaller\\hooks\\rthooks\\pyi_rth_inspect.py',
|
||||||
|
'PYSOURCE'),
|
||||||
|
('pyi_rth_pkgutil',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\PyInstaller\\hooks\\rthooks\\pyi_rth_pkgutil.py',
|
||||||
|
'PYSOURCE'),
|
||||||
|
('pyi_rth_multiprocessing',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\PyInstaller\\hooks\\rthooks\\pyi_rth_multiprocessing.py',
|
||||||
|
'PYSOURCE'),
|
||||||
|
('create_a_link', 'D:\\Projects\\cjgc_upload\\create_a_link.py', 'PYSOURCE'),
|
||||||
|
('selenium\\webdriver\\common\\windows\\selenium-manager.exe',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\windows\\selenium-manager.exe',
|
||||||
|
'BINARY'),
|
||||||
|
('python312.dll', 'C:\\Program Files\\Python312\\python312.dll', 'BINARY'),
|
||||||
|
('_multiprocessing.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_multiprocessing.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('pyexpat.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\pyexpat.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('_ssl.pyd', 'C:\\Program Files\\Python312\\DLLs\\_ssl.pyd', 'EXTENSION'),
|
||||||
|
('_hashlib.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_hashlib.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('unicodedata.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\unicodedata.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('_decimal.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_decimal.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('_socket.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_socket.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('select.pyd', 'C:\\Program Files\\Python312\\DLLs\\select.pyd', 'EXTENSION'),
|
||||||
|
('_lzma.pyd', 'C:\\Program Files\\Python312\\DLLs\\_lzma.pyd', 'EXTENSION'),
|
||||||
|
('_bz2.pyd', 'C:\\Program Files\\Python312\\DLLs\\_bz2.pyd', 'EXTENSION'),
|
||||||
|
('_ctypes.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_ctypes.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('_queue.pyd', 'C:\\Program Files\\Python312\\DLLs\\_queue.pyd', 'EXTENSION'),
|
||||||
|
('_overlapped.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_overlapped.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('_asyncio.pyd',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\_asyncio.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('_wmi.pyd', 'C:\\Program Files\\Python312\\DLLs\\_wmi.pyd', 'EXTENSION'),
|
||||||
|
('_uuid.pyd', 'C:\\Program Files\\Python312\\DLLs\\_uuid.pyd', 'EXTENSION'),
|
||||||
|
('charset_normalizer\\md__mypyc.cp312-win_amd64.pyd',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\charset_normalizer\\md__mypyc.cp312-win_amd64.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('charset_normalizer\\md.cp312-win_amd64.pyd',
|
||||||
|
'C:\\Program '
|
||||||
|
'Files\\Python312\\Lib\\site-packages\\charset_normalizer\\md.cp312-win_amd64.pyd',
|
||||||
|
'EXTENSION'),
|
||||||
|
('api-ms-win-core-synch-l1-2-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-synch-l1-2-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-conio-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-conio-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-environment-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-environment-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-convert-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-convert-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-time-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-time-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-string-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-string-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-filesystem-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-filesystem-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-locale-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-locale-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-process-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-process-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-math-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-math-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-heap-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-heap-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-runtime-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-runtime-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('VCRUNTIME140.dll',
|
||||||
|
'C:\\Program Files\\Python312\\VCRUNTIME140.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-stdio-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-stdio-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-crt-utility-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-crt-utility-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('libcrypto-3.dll',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\libcrypto-3.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('libssl-3.dll',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\libssl-3.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('libffi-8.dll',
|
||||||
|
'C:\\Program Files\\Python312\\DLLs\\libffi-8.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('ucrtbase.dll', 'D:\\Resource\\JDK\\JavaJDK17\\bin\\ucrtbase.dll', 'BINARY'),
|
||||||
|
('api-ms-win-core-synch-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-synch-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-sysinfo-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-sysinfo-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-processthreads-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-processthreads-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-string-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-string-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-timezone-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-timezone-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-console-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-console-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-heap-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-heap-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-memory-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-memory-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-processenvironment-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-processenvironment-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-errorhandling-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-errorhandling-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-datetime-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-datetime-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-file-l1-2-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-file-l1-2-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-processthreads-l1-1-1.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-processthreads-l1-1-1.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-rtlsupport-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-rtlsupport-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-interlocked-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-interlocked-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-namedpipe-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-namedpipe-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-file-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-file-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-handle-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-handle-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-profile-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-profile-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-file-l2-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-file-l2-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-libraryloader-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-libraryloader-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-util-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-util-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-localization-l1-2-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-localization-l1-2-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('api-ms-win-core-debug-l1-1-0.dll',
|
||||||
|
'D:\\Resource\\JDK\\JavaJDK17\\bin\\api-ms-win-core-debug-l1-1-0.dll',
|
||||||
|
'BINARY'),
|
||||||
|
('selenium\\webdriver\\common\\devtools\\v139\\py.typed',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\devtools\\v139\\py.typed',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\remote\\isDisplayed.js',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\remote\\isDisplayed.js',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\common\\linux\\selenium-manager',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\linux\\selenium-manager',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\common\\macos\\selenium-manager',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\macos\\selenium-manager',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\firefox\\webdriver_prefs.json',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\firefox\\webdriver_prefs.json',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\common\\devtools\\v138\\py.typed',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\devtools\\v138\\py.typed',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\common\\devtools\\v137\\py.typed',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\devtools\\v137\\py.typed',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\remote\\findElements.js',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\remote\\findElements.js',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\common\\mutation-listener.js',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\common\\mutation-listener.js',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\webdriver\\remote\\getAttribute.js',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\webdriver\\remote\\getAttribute.js',
|
||||||
|
'DATA'),
|
||||||
|
('selenium\\py.typed',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\selenium\\py.typed',
|
||||||
|
'DATA'),
|
||||||
|
('certifi\\cacert.pem',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\certifi\\cacert.pem',
|
||||||
|
'DATA'),
|
||||||
|
('certifi\\py.typed',
|
||||||
|
'C:\\Users\\ADMIN\\AppData\\Roaming\\Python\\Python312\\site-packages\\certifi\\py.typed',
|
||||||
|
'DATA'),
|
||||||
|
('base_library.zip',
|
||||||
|
'D:\\Projects\\cjgc_upload\\build\\create_a_link\\base_library.zip',
|
||||||
|
'DATA')],
|
||||||
|
'python312.dll',
|
||||||
|
False,
|
||||||
|
False,
|
||||||
|
False,
|
||||||
|
[],
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
None)
|
||||||
BIN
build/create_a_link/PYZ-00.pyz
Normal file
BIN
build/create_a_link/PYZ-00.pyz
Normal file
Binary file not shown.
1765
build/create_a_link/PYZ-00.toc
Normal file
1765
build/create_a_link/PYZ-00.toc
Normal file
File diff suppressed because it is too large
Load Diff
BIN
build/create_a_link/base_library.zip
Normal file
BIN
build/create_a_link/base_library.zip
Normal file
Binary file not shown.
BIN
build/create_a_link/create_a_link.pkg
Normal file
BIN
build/create_a_link/create_a_link.pkg
Normal file
Binary file not shown.
BIN
build/create_a_link/localpycs/pyimod01_archive.pyc
Normal file
BIN
build/create_a_link/localpycs/pyimod01_archive.pyc
Normal file
Binary file not shown.
BIN
build/create_a_link/localpycs/pyimod02_importers.pyc
Normal file
BIN
build/create_a_link/localpycs/pyimod02_importers.pyc
Normal file
Binary file not shown.
BIN
build/create_a_link/localpycs/pyimod03_ctypes.pyc
Normal file
BIN
build/create_a_link/localpycs/pyimod03_ctypes.pyc
Normal file
Binary file not shown.
BIN
build/create_a_link/localpycs/pyimod04_pywin32.pyc
Normal file
BIN
build/create_a_link/localpycs/pyimod04_pywin32.pyc
Normal file
Binary file not shown.
BIN
build/create_a_link/localpycs/struct.pyc
Normal file
BIN
build/create_a_link/localpycs/struct.pyc
Normal file
Binary file not shown.
64
build/create_a_link/warn-create_a_link.txt
Normal file
64
build/create_a_link/warn-create_a_link.txt
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
|
||||||
|
This file lists modules PyInstaller was not able to find. This does not
|
||||||
|
necessarily mean this module is required for running your program. Python and
|
||||||
|
Python 3rd-party packages include a lot of conditional or optional modules. For
|
||||||
|
example the module 'ntpath' only exists on Windows, whereas the module
|
||||||
|
'posixpath' only exists on Posix systems.
|
||||||
|
|
||||||
|
Types if import:
|
||||||
|
* top-level: imported at the top-level - look at these first
|
||||||
|
* conditional: imported within an if-statement
|
||||||
|
* delayed: imported within a function
|
||||||
|
* optional: imported within a try-except-statement
|
||||||
|
|
||||||
|
IMPORTANT: Do NOT post this list to the issue-tracker. Use it as a basis for
|
||||||
|
tracking down the missing module yourself. Thanks!
|
||||||
|
|
||||||
|
missing module named _posixshmem - imported by multiprocessing.resource_tracker (conditional), multiprocessing.shared_memory (conditional)
|
||||||
|
missing module named _scproxy - imported by urllib.request (conditional)
|
||||||
|
missing module named termios - imported by getpass (optional)
|
||||||
|
missing module named pwd - imported by posixpath (delayed, conditional, optional), shutil (delayed, optional), tarfile (optional), pathlib (delayed, optional), subprocess (delayed, conditional, optional), netrc (delayed, conditional), getpass (delayed)
|
||||||
|
missing module named _sha512 - imported by random (optional)
|
||||||
|
missing module named multiprocessing.BufferTooShort - imported by multiprocessing (top-level), multiprocessing.connection (top-level)
|
||||||
|
missing module named multiprocessing.AuthenticationError - imported by multiprocessing (top-level), multiprocessing.connection (top-level)
|
||||||
|
missing module named _posixsubprocess - imported by subprocess (conditional), multiprocessing.util (delayed)
|
||||||
|
missing module named _frozen_importlib_external - imported by importlib._bootstrap (delayed), importlib (optional), importlib.abc (optional), zipimport (top-level)
|
||||||
|
excluded module named _frozen_importlib - imported by importlib (optional), importlib.abc (optional), zipimport (top-level)
|
||||||
|
missing module named grp - imported by shutil (delayed, optional), tarfile (optional), pathlib (delayed, optional), subprocess (delayed, conditional, optional)
|
||||||
|
missing module named posix - imported by os (conditional, optional), shutil (conditional), importlib._bootstrap_external (conditional), posixpath (optional)
|
||||||
|
missing module named resource - imported by posix (top-level)
|
||||||
|
missing module named multiprocessing.get_context - imported by multiprocessing (top-level), multiprocessing.pool (top-level), multiprocessing.managers (top-level), multiprocessing.sharedctypes (top-level)
|
||||||
|
missing module named multiprocessing.TimeoutError - imported by multiprocessing (top-level), multiprocessing.pool (top-level)
|
||||||
|
missing module named multiprocessing.set_start_method - imported by multiprocessing (top-level), multiprocessing.spawn (top-level)
|
||||||
|
missing module named multiprocessing.get_start_method - imported by multiprocessing (top-level), multiprocessing.spawn (top-level)
|
||||||
|
missing module named pyimod02_importers - imported by C:\Program Files\Python312\Lib\site-packages\PyInstaller\hooks\rthooks\pyi_rth_pkgutil.py (delayed)
|
||||||
|
missing module named collections.Callable - imported by collections (optional), socks (optional)
|
||||||
|
missing module named zstandard - imported by urllib3.util.request (optional), urllib3.response (optional)
|
||||||
|
missing module named compression - imported by urllib3.util.request (optional), urllib3.response (optional)
|
||||||
|
missing module named brotli - imported by urllib3.util.request (optional), urllib3.response (optional)
|
||||||
|
missing module named brotlicffi - imported by urllib3.util.request (optional), urllib3.response (optional)
|
||||||
|
missing module named asyncio.DefaultEventLoopPolicy - imported by asyncio (delayed, conditional), asyncio.events (delayed, conditional)
|
||||||
|
missing module named annotationlib - imported by typing_extensions (conditional)
|
||||||
|
missing module named 'h2.events' - imported by urllib3.http2.connection (top-level)
|
||||||
|
missing module named 'h2.connection' - imported by urllib3.http2.connection (top-level)
|
||||||
|
missing module named h2 - imported by urllib3.http2.connection (top-level)
|
||||||
|
missing module named 'OpenSSL.crypto' - imported by urllib3.contrib.pyopenssl (delayed, conditional)
|
||||||
|
missing module named 'cryptography.x509' - imported by urllib3.contrib.pyopenssl (delayed, optional)
|
||||||
|
missing module named cryptography - imported by urllib3.contrib.pyopenssl (top-level), requests (conditional, optional)
|
||||||
|
missing module named OpenSSL - imported by urllib3.contrib.pyopenssl (top-level)
|
||||||
|
missing module named 'pyodide.ffi' - imported by urllib3.contrib.emscripten.fetch (delayed, optional)
|
||||||
|
missing module named pyodide - imported by urllib3.contrib.emscripten.fetch (top-level)
|
||||||
|
missing module named js - imported by urllib3.contrib.emscripten.fetch (top-level)
|
||||||
|
missing module named vms_lib - imported by platform (delayed, optional)
|
||||||
|
missing module named 'java.lang' - imported by platform (delayed, optional)
|
||||||
|
missing module named java - imported by platform (delayed)
|
||||||
|
missing module named _winreg - imported by platform (delayed, optional), selenium.webdriver.firefox.firefox_binary (delayed, optional)
|
||||||
|
missing module named wsaccel - imported by websocket._utils (optional)
|
||||||
|
missing module named 'python_socks.sync' - imported by websocket._http (optional)
|
||||||
|
missing module named 'python_socks._types' - imported by websocket._http (optional)
|
||||||
|
missing module named python_socks - imported by websocket._http (optional)
|
||||||
|
missing module named 'wsaccel.xormask' - imported by websocket._abnf (optional)
|
||||||
|
missing module named win_inet_pton - imported by socks (conditional, optional)
|
||||||
|
missing module named simplejson - imported by requests.compat (conditional, optional)
|
||||||
|
missing module named dummy_threading - imported by requests.cookies (optional)
|
||||||
|
missing module named fcntl - imported by subprocess (optional)
|
||||||
21568
build/create_a_link/xref-create_a_link.html
Normal file
21568
build/create_a_link/xref-create_a_link.html
Normal file
File diff suppressed because it is too large
Load Diff
13638
build/scheduler_two/Analysis-00.toc
Normal file
13638
build/scheduler_two/Analysis-00.toc
Normal file
File diff suppressed because it is too large
Load Diff
6808
build/scheduler_two/EXE-00.toc
Normal file
6808
build/scheduler_two/EXE-00.toc
Normal file
File diff suppressed because it is too large
Load Diff
6784
build/scheduler_two/PKG-00.toc
Normal file
6784
build/scheduler_two/PKG-00.toc
Normal file
File diff suppressed because it is too large
Load Diff
BIN
build/scheduler_two/PYZ-00.pyz
Normal file
BIN
build/scheduler_two/PYZ-00.pyz
Normal file
Binary file not shown.
6873
build/scheduler_two/PYZ-00.toc
Normal file
6873
build/scheduler_two/PYZ-00.toc
Normal file
File diff suppressed because it is too large
Load Diff
BIN
build/scheduler_two/base_library.zip
Normal file
BIN
build/scheduler_two/base_library.zip
Normal file
Binary file not shown.
BIN
build/scheduler_two/localpycs/pyimod01_archive.pyc
Normal file
BIN
build/scheduler_two/localpycs/pyimod01_archive.pyc
Normal file
Binary file not shown.
BIN
build/scheduler_two/localpycs/pyimod02_importers.pyc
Normal file
BIN
build/scheduler_two/localpycs/pyimod02_importers.pyc
Normal file
Binary file not shown.
BIN
build/scheduler_two/localpycs/pyimod03_ctypes.pyc
Normal file
BIN
build/scheduler_two/localpycs/pyimod03_ctypes.pyc
Normal file
Binary file not shown.
BIN
build/scheduler_two/localpycs/pyimod04_pywin32.pyc
Normal file
BIN
build/scheduler_two/localpycs/pyimod04_pywin32.pyc
Normal file
Binary file not shown.
BIN
build/scheduler_two/localpycs/struct.pyc
Normal file
BIN
build/scheduler_two/localpycs/struct.pyc
Normal file
Binary file not shown.
BIN
build/scheduler_two/scheduler_two.pkg
Normal file
BIN
build/scheduler_two/scheduler_two.pkg
Normal file
Binary file not shown.
347
build/scheduler_two/warn-scheduler_two.txt
Normal file
347
build/scheduler_two/warn-scheduler_two.txt
Normal file
@@ -0,0 +1,347 @@
|
|||||||
|
|
||||||
|
This file lists modules PyInstaller was not able to find. This does not
|
||||||
|
necessarily mean this module is required for running your program. Python and
|
||||||
|
Python 3rd-party packages include a lot of conditional or optional modules. For
|
||||||
|
example the module 'ntpath' only exists on Windows, whereas the module
|
||||||
|
'posixpath' only exists on Posix systems.
|
||||||
|
|
||||||
|
Types if import:
|
||||||
|
* top-level: imported at the top-level - look at these first
|
||||||
|
* conditional: imported within an if-statement
|
||||||
|
* delayed: imported within a function
|
||||||
|
* optional: imported within a try-except-statement
|
||||||
|
|
||||||
|
IMPORTANT: Do NOT post this list to the issue-tracker. Use it as a basis for
|
||||||
|
tracking down the missing module yourself. Thanks!
|
||||||
|
|
||||||
|
missing module named pwd - imported by shutil (delayed, optional), tarfile (optional), pathlib (delayed, optional), subprocess (delayed, conditional, optional), posixpath (delayed, conditional, optional), netrc (delayed, conditional), getpass (delayed), http.server (delayed, optional), psutil (optional), setuptools._distutils.util (delayed, conditional, optional), setuptools._distutils.archive_util (optional), setuptools._vendor.backports.tarfile (optional)
|
||||||
|
missing module named grp - imported by shutil (delayed, optional), tarfile (optional), pathlib (delayed, optional), subprocess (delayed, conditional, optional), setuptools._distutils.archive_util (optional), setuptools._vendor.backports.tarfile (optional)
|
||||||
|
missing module named _posixsubprocess - imported by subprocess (conditional), multiprocessing.util (delayed)
|
||||||
|
missing module named fcntl - imported by subprocess (optional)
|
||||||
|
missing module named _posixshmem - imported by multiprocessing.resource_tracker (conditional), multiprocessing.shared_memory (conditional)
|
||||||
|
missing module named _scproxy - imported by urllib.request (conditional)
|
||||||
|
missing module named termios - imported by getpass (optional), tty (top-level)
|
||||||
|
missing module named _sha512 - imported by random (optional)
|
||||||
|
missing module named urllib.urlopen - imported by urllib (delayed, optional), lxml.html (delayed, optional)
|
||||||
|
missing module named urllib.urlencode - imported by urllib (delayed, optional), lxml.html (delayed, optional)
|
||||||
|
missing module named multiprocessing.BufferTooShort - imported by multiprocessing (top-level), multiprocessing.connection (top-level)
|
||||||
|
missing module named multiprocessing.AuthenticationError - imported by multiprocessing (top-level), multiprocessing.connection (top-level)
|
||||||
|
missing module named _frozen_importlib_external - imported by importlib._bootstrap (delayed), importlib (optional), importlib.abc (optional), zipimport (top-level)
|
||||||
|
excluded module named _frozen_importlib - imported by importlib (optional), importlib.abc (optional), zipimport (top-level)
|
||||||
|
missing module named posix - imported by shutil (conditional), importlib._bootstrap_external (conditional), os (conditional, optional), posixpath (optional)
|
||||||
|
missing module named resource - imported by posix (top-level)
|
||||||
|
missing module named multiprocessing.get_context - imported by multiprocessing (top-level), multiprocessing.pool (top-level), multiprocessing.managers (top-level), multiprocessing.sharedctypes (top-level)
|
||||||
|
missing module named multiprocessing.TimeoutError - imported by multiprocessing (top-level), multiprocessing.pool (top-level)
|
||||||
|
missing module named multiprocessing.set_start_method - imported by multiprocessing (top-level), multiprocessing.spawn (top-level)
|
||||||
|
missing module named multiprocessing.get_start_method - imported by multiprocessing (top-level), multiprocessing.spawn (top-level)
|
||||||
|
missing module named asyncio.DefaultEventLoopPolicy - imported by asyncio (delayed, conditional), asyncio.events (delayed, conditional)
|
||||||
|
missing module named annotationlib - imported by typing_extensions (conditional)
|
||||||
|
missing module named vms_lib - imported by platform (delayed, optional)
|
||||||
|
missing module named 'java.lang' - imported by platform (delayed, optional)
|
||||||
|
missing module named java - imported by platform (delayed)
|
||||||
|
missing module named _winreg - imported by platform (delayed, optional), selenium.webdriver.firefox.firefox_binary (delayed, optional)
|
||||||
|
missing module named usercustomize - imported by site (delayed, optional)
|
||||||
|
missing module named sitecustomize - imported by site (delayed, optional)
|
||||||
|
missing module named readline - imported by cmd (delayed, conditional, optional), code (delayed, conditional, optional), pdb (delayed, optional), site (delayed, optional), rlcompleter (optional)
|
||||||
|
missing module named _typeshed - imported by setuptools._distutils.dist (conditional), setuptools.glob (conditional), setuptools.compat.py311 (conditional), pkg_resources (conditional)
|
||||||
|
missing module named '_typeshed.importlib' - imported by pkg_resources (conditional)
|
||||||
|
missing module named jnius - imported by setuptools._vendor.platformdirs.android (delayed, conditional, optional)
|
||||||
|
missing module named android - imported by setuptools._vendor.platformdirs.android (delayed, conditional, optional)
|
||||||
|
missing module named importlib_resources - imported by setuptools._vendor.jaraco.text (optional)
|
||||||
|
missing module named jaraco.text.yield_lines - imported by setuptools._vendor.jaraco.text (top-level), setuptools._entry_points (top-level), setuptools.command._requirestxt (top-level)
|
||||||
|
missing module named _manylinux - imported by packaging._manylinux (delayed, optional), setuptools._vendor.packaging._manylinux (delayed, optional), setuptools._vendor.wheel.vendored.packaging._manylinux (delayed, optional)
|
||||||
|
missing module named trove_classifiers - imported by setuptools.config._validate_pyproject.formats (optional)
|
||||||
|
missing module named pyimod02_importers - imported by C:\Program Files\Python312\Lib\site-packages\PyInstaller\hooks\rthooks\pyi_rth_pkgutil.py (delayed), C:\Program Files\Python312\Lib\site-packages\PyInstaller\hooks\rthooks\pyi_rth_pkgres.py (delayed)
|
||||||
|
missing module named collections.Mapping - imported by collections (optional), pytz.lazy (optional)
|
||||||
|
missing module named collections.Callable - imported by collections (optional), socks (optional)
|
||||||
|
missing module named simplejson - imported by requests.compat (conditional, optional)
|
||||||
|
missing module named dummy_threading - imported by requests.cookies (optional)
|
||||||
|
missing module named zstandard - imported by urllib3.util.request (optional), urllib3.response (optional)
|
||||||
|
missing module named compression - imported by urllib3.util.request (optional), urllib3.response (optional)
|
||||||
|
missing module named 'h2.events' - imported by urllib3.http2.connection (top-level)
|
||||||
|
missing module named 'h2.connection' - imported by urllib3.http2.connection (top-level)
|
||||||
|
missing module named h2 - imported by urllib3.http2.connection (top-level)
|
||||||
|
missing module named brotli - imported by urllib3.util.request (optional), urllib3.response (optional)
|
||||||
|
missing module named brotlicffi - imported by urllib3.util.request (optional), urllib3.response (optional)
|
||||||
|
missing module named win_inet_pton - imported by socks (conditional, optional)
|
||||||
|
missing module named cryptography - imported by urllib3.contrib.pyopenssl (top-level), requests (conditional, optional)
|
||||||
|
missing module named 'OpenSSL.crypto' - imported by urllib3.contrib.pyopenssl (delayed, conditional)
|
||||||
|
missing module named 'cryptography.x509' - imported by urllib3.contrib.pyopenssl (delayed, optional)
|
||||||
|
missing module named OpenSSL - imported by urllib3.contrib.pyopenssl (top-level)
|
||||||
|
missing module named 'pyodide.ffi' - imported by urllib3.contrib.emscripten.fetch (delayed, optional)
|
||||||
|
missing module named pyodide - imported by urllib3.contrib.emscripten.fetch (top-level)
|
||||||
|
missing module named js - imported by urllib3.contrib.emscripten.fetch (top-level)
|
||||||
|
missing module named six.moves.range - imported by six.moves (top-level), dateutil.rrule (top-level)
|
||||||
|
runtime module named six.moves - imported by dateutil.tz.tz (top-level), dateutil.tz._factories (top-level), dateutil.tz.win (top-level), dateutil.rrule (top-level)
|
||||||
|
missing module named dateutil.tz.tzfile - imported by dateutil.tz (top-level), dateutil.zoneinfo (top-level)
|
||||||
|
missing module named StringIO - imported by six (conditional)
|
||||||
|
missing module named numexpr - imported by pandas.core.computation.expressions (conditional), pandas.core.computation.engines (delayed)
|
||||||
|
missing module named numba - imported by pandas.core._numba.executor (delayed, conditional), pandas.core.util.numba_ (delayed, conditional), pandas.core.window.numba_ (delayed, conditional), pandas.core.window.online (delayed, conditional), pandas.core._numba.kernels.mean_ (top-level), pandas.core._numba.kernels.shared (top-level), pandas.core._numba.kernels.sum_ (top-level), pandas.core._numba.kernels.min_max_ (top-level), pandas.core._numba.kernels.var_ (top-level), pandas.core.groupby.numba_ (delayed, conditional), pandas.core._numba.extensions (top-level)
|
||||||
|
missing module named 'numba.extending' - imported by pandas.core._numba.kernels.sum_ (top-level)
|
||||||
|
missing module named 'pyarrow.compute' - imported by pandas.core.arrays.arrow.accessors (conditional), pandas.core.arrays._arrow_string_mixins (conditional), pandas.core.arrays.string_arrow (conditional), pandas.core.reshape.merge (delayed, conditional), pandas.core.arrays.arrow.array (conditional)
|
||||||
|
missing module named 'numba.typed' - imported by pandas.core._numba.extensions (delayed)
|
||||||
|
missing module named 'numba.core' - imported by pandas.core._numba.extensions (top-level)
|
||||||
|
missing module named pyarrow - imported by pandas.core.arrays.arrow.accessors (conditional), pandas.core.arrays.masked (delayed), pandas.core.arrays.boolean (delayed, conditional), pandas.core.arrays.string_ (delayed, conditional), pandas.core.arrays._arrow_string_mixins (conditional), pandas.core.arrays.string_arrow (conditional), pandas.core.arrays.arrow._arrow_utils (top-level), pandas.core.interchange.utils (delayed, conditional), pandas.core.strings.accessor (delayed, conditional), pandas.io.parsers.base_parser (delayed, conditional), pandas.core.arrays.interval (delayed), pandas.core.arrays.arrow.extension_types (top-level), pandas.core.arrays.period (delayed), pandas.core.methods.describe (delayed, conditional), pandas.io.sql (delayed, conditional), pandas.core.reshape.merge (delayed, conditional), pandas.core.arrays.numeric (delayed, conditional), pandas.core.interchange.buffer (conditional), pandas.io.feather_format (delayed), pandas.core.indexes.base (delayed, conditional), pandas.core.dtypes.cast (delayed, conditional), pandas.core.arrays.arrow.array (conditional), pandas.core.dtypes.dtypes (delayed, conditional), pandas.compat.pyarrow (optional), pandas.core.reshape.encoding (delayed, conditional), pandas._testing (conditional)
|
||||||
|
missing module named _dummy_thread - imported by numpy._core.arrayprint (optional)
|
||||||
|
missing module named numpy._core.void - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.vecmat - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.ushort - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.unsignedinteger - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.ulonglong - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.ulong - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.uintp - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.uintc - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.uint64 - imported by numpy._core (conditional), numpy (conditional), numpy._array_api_info (top-level)
|
||||||
|
missing module named numpy._core.uint32 - imported by numpy._core (conditional), numpy (conditional), numpy._array_api_info (top-level)
|
||||||
|
missing module named numpy._core.uint16 - imported by numpy._core (conditional), numpy (conditional), numpy._array_api_info (top-level)
|
||||||
|
missing module named numpy._core.uint - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.ubyte - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.trunc - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.true_divide - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.timedelta64 - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.tanh - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.tan - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.subtract - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.str_ - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.square - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.spacing - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.sinh - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.signedinteger - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.short - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.rint - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.right_shift - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.remainder - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.radians - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.rad2deg - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.power - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.positive - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.pi - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.not_equal - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.negative - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.modf - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.mod - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.minimum - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.maximum - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.matvec - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.longdouble - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.long - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.logical_xor - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.logical_or - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.logical_not - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.logical_and - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.logaddexp2 - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.logaddexp - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.log2 - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.log1p - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.log - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.less_equal - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.less - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.left_shift - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.ldexp - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.lcm - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.integer - imported by numpy._core (conditional), numpy (conditional), numpy.fft._helper (top-level)
|
||||||
|
missing module named numpy._core.int8 - imported by numpy._core (conditional), numpy (conditional), numpy._array_api_info (top-level)
|
||||||
|
missing module named numpy._core.int64 - imported by numpy._core (conditional), numpy (conditional), numpy._array_api_info (top-level)
|
||||||
|
missing module named numpy._core.int32 - imported by numpy._core (conditional), numpy (conditional), numpy._array_api_info (top-level)
|
||||||
|
missing module named numpy._core.int16 - imported by numpy._core (conditional), numpy (conditional), numpy._array_api_info (top-level)
|
||||||
|
missing module named numpy._core.hypot - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.heaviside - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.half - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.greater_equal - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.greater - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.gcd - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.frompyfunc - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.frexp - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.fmod - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.fmin - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.fmax - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.floor_divide - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.floor - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.floating - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.float_power - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.float16 - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.fabs - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.expm1 - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.exp - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.euler_gamma - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.equal - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.e - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.divmod - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.degrees - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.deg2rad - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.datetime64 - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.cosh - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.cos - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.copysign - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.conjugate - imported by numpy._core (conditional), numpy (conditional), numpy.fft._pocketfft (top-level)
|
||||||
|
missing module named numpy._core.conj - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.complex64 - imported by numpy._core (conditional), numpy (conditional), numpy._array_api_info (top-level)
|
||||||
|
missing module named numpy._core.clongdouble - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.character - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.ceil - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.cbrt - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.bytes_ - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.byte - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.bool_ - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.bitwise_xor - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.bitwise_or - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.bitwise_count - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.bitwise_and - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.arctanh - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.arctan2 - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.arctan - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.arcsinh - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.arcsin - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.arccosh - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.arccos - imported by numpy._core (conditional), numpy (conditional)
|
||||||
|
missing module named numpy._core.ones - imported by numpy._core (top-level), numpy.lib._polynomial_impl (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.hstack - imported by numpy._core (top-level), numpy.lib._polynomial_impl (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.atleast_1d - imported by numpy._core (top-level), numpy.lib._polynomial_impl (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.atleast_3d - imported by numpy._core (top-level), numpy.lib._shape_base_impl (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.vstack - imported by numpy._core (top-level), numpy.lib._shape_base_impl (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.linspace - imported by numpy._core (top-level), numpy.lib._index_tricks_impl (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.result_type - imported by numpy._core (delayed), numpy.testing._private.utils (delayed), numpy (conditional), numpy.fft._pocketfft (top-level)
|
||||||
|
missing module named numpy._core.number - imported by numpy._core (delayed), numpy.testing._private.utils (delayed), numpy (conditional)
|
||||||
|
missing module named numpy._core.max - imported by numpy._core (delayed), numpy.testing._private.utils (delayed), numpy (conditional)
|
||||||
|
missing module named numpy._core.array2string - imported by numpy._core (delayed), numpy.testing._private.utils (delayed), numpy (conditional)
|
||||||
|
missing module named numpy._core.signbit - imported by numpy._core (delayed), numpy.testing._private.utils (delayed), numpy (conditional)
|
||||||
|
missing module named numpy._core.isscalar - imported by numpy._core (delayed), numpy.testing._private.utils (delayed), numpy.lib._polynomial_impl (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.isnat - imported by numpy._core (top-level), numpy.testing._private.utils (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.array_repr - imported by numpy._core (top-level), numpy.testing._private.utils (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.arange - imported by numpy._core (top-level), numpy.testing._private.utils (top-level), numpy (conditional), numpy.fft._helper (top-level)
|
||||||
|
missing module named numpy._core.float32 - imported by numpy._core (top-level), numpy.testing._private.utils (top-level), numpy (conditional), numpy._array_api_info (top-level)
|
||||||
|
missing module named numpy._core.vecdot - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.matrix_transpose - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.matmul - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.tensordot - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.outer - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.cross - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.trace - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.diagonal - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.reciprocal - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional), numpy.fft._pocketfft (top-level)
|
||||||
|
missing module named numpy._core.sort - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.argsort - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.sign - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.isnan - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy.testing._private.utils (delayed), numpy (conditional)
|
||||||
|
missing module named numpy._core.count_nonzero - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.divide - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.swapaxes - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.object_ - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy.testing._private.utils (delayed), numpy (conditional)
|
||||||
|
missing module named numpy._core.asanyarray - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.intp - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy.testing._private.utils (top-level), numpy (conditional), numpy._array_api_info (top-level)
|
||||||
|
missing module named numpy._core.atleast_2d - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.prod - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.amax - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.amin - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.moveaxis - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.errstate - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy.testing._private.utils (delayed), numpy (conditional)
|
||||||
|
missing module named numpy._core.finfo - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy.lib._polynomial_impl (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.isfinite - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.sum - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.sqrt - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional), numpy.fft._pocketfft (top-level)
|
||||||
|
missing module named numpy._core.multiply - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.add - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.dot - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy.lib._polynomial_impl (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.inf - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy.testing._private.utils (delayed), numpy (conditional)
|
||||||
|
missing module named numpy._core.all - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy.testing._private.utils (delayed), numpy (conditional)
|
||||||
|
missing module named numpy._core.newaxis - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.complexfloating - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.inexact - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.cdouble - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.csingle - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.double - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.single - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.intc - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.empty_like - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional), numpy.fft._pocketfft (top-level)
|
||||||
|
missing module named numpy._core.empty - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy.testing._private.utils (top-level), numpy (conditional), numpy.fft._helper (top-level)
|
||||||
|
missing module named numpy._core.zeros - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.array - imported by numpy._core (top-level), numpy.linalg._linalg (top-level), numpy.testing._private.utils (top-level), numpy.lib._polynomial_impl (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.iinfo - imported by numpy._core (top-level), numpy.lib._twodim_base_impl (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.transpose - imported by numpy._core (top-level), numpy.lib._function_base_impl (top-level), numpy.linalg._linalg (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.ndarray - imported by numpy._core (top-level), numpy.lib._utils_impl (top-level), numpy.testing._private.utils (top-level), numpy (conditional)
|
||||||
|
missing module named numpy._core.asarray - imported by numpy._core (top-level), numpy.lib._array_utils_impl (top-level), numpy.linalg._linalg (top-level), numpy (conditional), numpy.fft._pocketfft (top-level), numpy.fft._helper (top-level)
|
||||||
|
missing module named threadpoolctl - imported by numpy.lib._utils_impl (delayed, optional)
|
||||||
|
missing module named 'scipy.stats' - imported by pandas.core.nanops (delayed, conditional), pandas.plotting._matplotlib.misc (delayed, conditional), pandas.plotting._matplotlib.hist (delayed)
|
||||||
|
missing module named scipy - imported by pandas.core.nanops (delayed, conditional), pandas.core.missing (delayed)
|
||||||
|
missing module named traitlets - imported by pandas.io.formats.printing (delayed, conditional)
|
||||||
|
missing module named 'IPython.core' - imported by pandas.io.formats.printing (delayed, conditional)
|
||||||
|
missing module named IPython - imported by pandas.io.formats.printing (delayed)
|
||||||
|
missing module named jinja2 - imported by pyparsing.diagram (top-level), pandas.io.formats.style (top-level)
|
||||||
|
missing module named defusedxml - imported by PIL.Image (optional), openpyxl.xml (delayed, optional)
|
||||||
|
missing module named 'defusedxml.ElementTree' - imported by openpyxl.xml.functions (conditional)
|
||||||
|
missing module named htmlentitydefs - imported by lxml.html.soupparser (optional)
|
||||||
|
missing module named BeautifulSoup - imported by lxml.html.soupparser (optional)
|
||||||
|
missing module named cchardet - imported by bs4.dammit (optional)
|
||||||
|
missing module named 'html5lib.treebuilders' - imported by bs4.builder._html5lib (top-level), lxml.html._html5builder (top-level), lxml.html.html5parser (top-level)
|
||||||
|
missing module named 'html5lib.constants' - imported by bs4.builder._html5lib (top-level)
|
||||||
|
missing module named html5lib - imported by bs4.builder._html5lib (top-level), lxml.html.html5parser (top-level)
|
||||||
|
missing module named urlparse - imported by lxml.ElementInclude (optional), lxml.html.html5parser (optional)
|
||||||
|
missing module named urllib2 - imported by lxml.ElementInclude (optional), lxml.html.html5parser (optional)
|
||||||
|
missing module named lxml_html_clean - imported by lxml.html.clean (optional)
|
||||||
|
missing module named cssselect - imported by lxml.cssselect (optional)
|
||||||
|
missing module named olefile - imported by PIL.FpxImagePlugin (top-level), PIL.MicImagePlugin (top-level)
|
||||||
|
missing module named openpyxl.tests - imported by openpyxl.reader.excel (optional)
|
||||||
|
missing module named 'odf.config' - imported by pandas.io.excel._odswriter (delayed)
|
||||||
|
missing module named 'odf.style' - imported by pandas.io.excel._odswriter (delayed)
|
||||||
|
missing module named 'odf.text' - imported by pandas.io.excel._odfreader (delayed), pandas.io.excel._odswriter (delayed)
|
||||||
|
missing module named 'odf.table' - imported by pandas.io.excel._odfreader (delayed), pandas.io.excel._odswriter (delayed)
|
||||||
|
missing module named 'odf.opendocument' - imported by pandas.io.excel._odfreader (delayed), pandas.io.excel._odswriter (delayed)
|
||||||
|
missing module named cStringIO - imported by xlrd.timemachine (conditional)
|
||||||
|
missing module named pyxlsb - imported by pandas.io.excel._pyxlsb (delayed, conditional)
|
||||||
|
missing module named 'odf.office' - imported by pandas.io.excel._odfreader (delayed)
|
||||||
|
missing module named 'odf.element' - imported by pandas.io.excel._odfreader (delayed)
|
||||||
|
missing module named 'odf.namespaces' - imported by pandas.io.excel._odfreader (delayed)
|
||||||
|
missing module named odf - imported by pandas.io.excel._odfreader (conditional)
|
||||||
|
missing module named python_calamine - imported by pandas.io.excel._calamine (delayed, conditional)
|
||||||
|
missing module named railroad - imported by pyparsing.diagram (top-level)
|
||||||
|
missing module named pyparsing.Word - imported by pyparsing (delayed), pyparsing.unicode (delayed)
|
||||||
|
missing module named 'tornado.template' - imported by matplotlib.backends.backend_webagg (delayed)
|
||||||
|
missing module named 'tornado.websocket' - imported by matplotlib.backends.backend_webagg (top-level)
|
||||||
|
missing module named 'tornado.ioloop' - imported by matplotlib.backends.backend_webagg (top-level)
|
||||||
|
missing module named 'tornado.web' - imported by matplotlib.backends.backend_webagg (top-level)
|
||||||
|
missing module named tornado - imported by matplotlib.backends.backend_webagg (optional), matplotlib.backends.backend_webagg_core (delayed)
|
||||||
|
missing module named gi - imported by matplotlib.cbook (delayed, conditional)
|
||||||
|
missing module named numpy.VisibleDeprecationWarning - imported by numpy (optional), matplotlib.cbook (optional)
|
||||||
|
missing module named setuptools_scm - imported by matplotlib (delayed, conditional, optional)
|
||||||
|
missing module named markupsafe - imported by pandas.io.formats.style_render (top-level)
|
||||||
|
missing module named botocore - imported by pandas.io.common (delayed, conditional, optional)
|
||||||
|
missing module named sets - imported by pytz.tzinfo (optional)
|
||||||
|
missing module named UserDict - imported by pytz.lazy (optional)
|
||||||
|
missing module named 'scipy.sparse' - imported by pandas.core.arrays.sparse.array (conditional), pandas.core.arrays.sparse.scipy_sparse (delayed, conditional), pandas.core.arrays.sparse.accessor (delayed), pandas.core.dtypes.common (delayed, conditional, optional)
|
||||||
|
missing module named pandas.core.internals.Block - imported by pandas.core.internals (conditional), pandas.io.pytables (conditional)
|
||||||
|
missing module named Foundation - imported by pandas.io.clipboard (delayed, conditional, optional)
|
||||||
|
missing module named AppKit - imported by pandas.io.clipboard (delayed, conditional, optional)
|
||||||
|
missing module named PyQt4 - imported by pandas.io.clipboard (delayed, conditional, optional)
|
||||||
|
missing module named qtpy - imported by pandas.io.clipboard (delayed, conditional, optional)
|
||||||
|
missing module named pysqlcipher3 - imported by sqlalchemy.dialects.sqlite.pysqlcipher (delayed)
|
||||||
|
missing module named sqlcipher3 - imported by sqlalchemy.dialects.sqlite.pysqlcipher (delayed, optional)
|
||||||
|
missing module named psycopg2 - imported by sqlalchemy.dialects.postgresql.psycopg2 (delayed)
|
||||||
|
missing module named 'psycopg.pq' - imported by sqlalchemy.dialects.postgresql.psycopg (delayed)
|
||||||
|
missing module named 'psycopg.types' - imported by sqlalchemy.dialects.postgresql.psycopg (delayed, conditional)
|
||||||
|
missing module named 'psycopg.adapt' - imported by sqlalchemy.dialects.postgresql.psycopg (delayed, conditional)
|
||||||
|
missing module named psycopg - imported by sqlalchemy.dialects.postgresql.psycopg (delayed, conditional)
|
||||||
|
missing module named asyncpg - imported by sqlalchemy.dialects.postgresql.asyncpg (delayed)
|
||||||
|
missing module named oracledb - imported by sqlalchemy.dialects.oracle.oracledb (delayed, conditional)
|
||||||
|
missing module named cx_Oracle - imported by sqlalchemy.dialects.oracle.cx_oracle (delayed)
|
||||||
|
missing module named 'mysql.connector' - imported by sqlalchemy.dialects.mysql.mysqlconnector (delayed, conditional, optional)
|
||||||
|
missing module named mysql - imported by sqlalchemy.dialects.mysql.mysqlconnector (delayed)
|
||||||
|
missing module named asyncmy - imported by sqlalchemy.dialects.mysql.asyncmy (delayed)
|
||||||
|
missing module named nacl - imported by pymysql._auth (delayed, optional)
|
||||||
|
missing module named 'cryptography.hazmat' - imported by pymysql._auth (optional)
|
||||||
|
missing module named tables - imported by pandas.io.pytables (delayed, conditional)
|
||||||
|
missing module named 'pyarrow.fs' - imported by pandas.io.orc (conditional)
|
||||||
|
missing module named fsspec - imported by pandas.io.orc (conditional)
|
||||||
|
missing module named 'pyarrow.parquet' - imported by pandas.io.parquet (delayed)
|
||||||
|
missing module named google - imported by pandas.io.gbq (conditional)
|
||||||
|
missing module named 'numpy_distutils.cpuinfo' - imported by numpy.f2py.diagnose (delayed, conditional, optional)
|
||||||
|
missing module named 'numpy_distutils.fcompiler' - imported by numpy.f2py.diagnose (delayed, conditional, optional)
|
||||||
|
missing module named 'numpy_distutils.command' - imported by numpy.f2py.diagnose (delayed, conditional, optional)
|
||||||
|
missing module named numpy_distutils - imported by numpy.f2py.diagnose (delayed, optional)
|
||||||
|
missing module named yaml - imported by numpy.__config__ (delayed)
|
||||||
|
missing module named numpy._distributor_init_local - imported by numpy (optional), numpy._distributor_init (optional)
|
||||||
|
missing module named pytest - imported by pandas._testing._io (delayed), pandas._testing (delayed)
|
||||||
|
missing module named wsaccel - imported by websocket._utils (optional)
|
||||||
|
missing module named 'python_socks.sync' - imported by websocket._http (optional)
|
||||||
|
missing module named 'python_socks._types' - imported by websocket._http (optional)
|
||||||
|
missing module named python_socks - imported by websocket._http (optional)
|
||||||
|
missing module named 'wsaccel.xormask' - imported by websocket._abnf (optional)
|
||||||
76457
build/scheduler_two/xref-scheduler_two.html
Normal file
76457
build/scheduler_two/xref-scheduler_two.html
Normal file
File diff suppressed because it is too large
Load Diff
347
create_a_link.py
Normal file
347
create_a_link.py
Normal file
@@ -0,0 +1,347 @@
|
|||||||
|
import subprocess
|
||||||
|
import re
|
||||||
|
import time
|
||||||
|
import requests
|
||||||
|
import json
|
||||||
|
from appium import webdriver
|
||||||
|
from appium.webdriver.common.appiumby import AppiumBy
|
||||||
|
from appium.options.android import UiAutomator2Options
|
||||||
|
from selenium.webdriver.support.ui import WebDriverWait
|
||||||
|
from selenium.webdriver.support import expected_conditions as EC
|
||||||
|
from urllib3.connection import port_by_scheme
|
||||||
|
|
||||||
|
|
||||||
|
# =======================
|
||||||
|
# 基础工具函数
|
||||||
|
# =======================
|
||||||
|
|
||||||
|
def run_command(command):
|
||||||
|
"""执行系统命令并返回输出"""
|
||||||
|
result = subprocess.run(command, shell=True, capture_output=True, text=True)
|
||||||
|
return result.stdout.strip()
|
||||||
|
|
||||||
|
|
||||||
|
# =======================
|
||||||
|
# API请求函数
|
||||||
|
# =======================
|
||||||
|
|
||||||
|
def get_new_port(yh_id):
|
||||||
|
"""从服务器获取新的端口号"""
|
||||||
|
url = "https://engineering.yuxindazhineng.com/index/index/get_yh_port"
|
||||||
|
headers = {
|
||||||
|
"Content-Type": "application/x-www-form-urlencoded"
|
||||||
|
}
|
||||||
|
data = {
|
||||||
|
"yh_id": yh_id
|
||||||
|
}
|
||||||
|
|
||||||
|
try:
|
||||||
|
print(f"🔍 查询服务器新端口号,用户ID: {yh_id}")
|
||||||
|
response = requests.post(url, headers=headers, data=data, timeout=10)
|
||||||
|
|
||||||
|
if response.status_code == 200:
|
||||||
|
result = response.json()
|
||||||
|
if result.get("code") == 0:
|
||||||
|
print(f"✅ 查询成功,新端口号: {result.get('data', '未知')}")
|
||||||
|
return result.get("data", None)
|
||||||
|
else:
|
||||||
|
print(f"❌ 查询失败: {result.get('msg', '未知错误')}")
|
||||||
|
return None
|
||||||
|
else:
|
||||||
|
print(f"❌ 服务器响应错误: {response.status_code}")
|
||||||
|
return None
|
||||||
|
except requests.exceptions.RequestException as e:
|
||||||
|
print(f"❌ 网络请求失败: {e}")
|
||||||
|
return None
|
||||||
|
|
||||||
|
def get_accounts_from_server(yh_id):
|
||||||
|
"""从服务器获取账户信息"""
|
||||||
|
url = "http://www.yuxindazhineng.com:3002/api/accounts/get_uplaod_data"
|
||||||
|
headers = {
|
||||||
|
"Content-Type": "application/json"
|
||||||
|
}
|
||||||
|
data = {
|
||||||
|
"yh_id": yh_id
|
||||||
|
}
|
||||||
|
|
||||||
|
try:
|
||||||
|
print(f"🔍 查询服务器账户信息,用户ID: {yh_id}")
|
||||||
|
response = requests.post(url, headers=headers, json=data, timeout=10)
|
||||||
|
|
||||||
|
if response.status_code == 200:
|
||||||
|
result = response.json()
|
||||||
|
if result.get("code") == 0:
|
||||||
|
print(f"✅ 查询成功,找到 {result.get('total', 0)} 个账户")
|
||||||
|
return result.get("data", [])
|
||||||
|
else:
|
||||||
|
print(f"❌ 查询失败: {result.get('message', '未知错误')}")
|
||||||
|
return []
|
||||||
|
else:
|
||||||
|
print(f"❌ 服务器响应错误: {response.status_code}")
|
||||||
|
return []
|
||||||
|
except requests.exceptions.RequestException as e:
|
||||||
|
print(f"❌ 网络请求失败: {e}")
|
||||||
|
return []
|
||||||
|
except json.JSONDecodeError as e:
|
||||||
|
print(f"❌ JSON解析失败: {e}")
|
||||||
|
return []
|
||||||
|
|
||||||
|
|
||||||
|
def update_device_info(account_id, device_name, device_port, device_ip):
|
||||||
|
"""更新设备信息到服务器"""
|
||||||
|
url = "http://www.yuxindazhineng.com:3002/api/accounts/update"
|
||||||
|
headers = {
|
||||||
|
"Content-Type": "application/json"
|
||||||
|
}
|
||||||
|
data = {
|
||||||
|
"account_id": str(account_id),
|
||||||
|
"account_data": {
|
||||||
|
"device_name": str(device_name),
|
||||||
|
"device_port": str(device_port),
|
||||||
|
"device_ip": str(device_ip)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try:
|
||||||
|
print(f"🔄 更新设备信息,账户ID: {account_id}")
|
||||||
|
print(f" 设备信息: 名称={device_name}, 端口={device_port}, IP={device_ip}")
|
||||||
|
|
||||||
|
response = requests.post(url, headers=headers, json=data, timeout=10)
|
||||||
|
|
||||||
|
if response.status_code == 200:
|
||||||
|
result = response.json()
|
||||||
|
if result.get("code") == 0:
|
||||||
|
print(f"✅ 更新成功: {result.get('message', '未知信息')}")
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
print(f"❌ 更新失败: {result.get('message', '未知错误')}")
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
print(f"❌ 服务器响应错误: {response.status_code}")
|
||||||
|
return False
|
||||||
|
except requests.exceptions.RequestException as e:
|
||||||
|
print(f"❌ 网络请求失败: {e}")
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
# =======================
|
||||||
|
# Appium 启动
|
||||||
|
# =======================
|
||||||
|
|
||||||
|
def start_appium():
|
||||||
|
print("🚀 启动 Appium Server ...")
|
||||||
|
subprocess.Popen(
|
||||||
|
["appium.cmd", "-a", "127.0.0.1", "-p", "4723"],
|
||||||
|
stdout=subprocess.DEVNULL,
|
||||||
|
stderr=subprocess.DEVNULL
|
||||||
|
)
|
||||||
|
time.sleep(5) # 给 Appium 启动时间
|
||||||
|
print("✅ Appium Server 已启动")
|
||||||
|
|
||||||
|
|
||||||
|
# =======================
|
||||||
|
# 启动沉降观测 App
|
||||||
|
# =======================
|
||||||
|
|
||||||
|
def start_settlement_app(device_id, device_ip, device_port):
|
||||||
|
print(f"📱 使用 Appium 连接设备: {device_id}")
|
||||||
|
|
||||||
|
options = UiAutomator2Options()
|
||||||
|
options.platform_name = "Android"
|
||||||
|
options.device_name = device_id
|
||||||
|
options.udid = device_id
|
||||||
|
|
||||||
|
# ⚠️ TODO:替换为你的真实信息
|
||||||
|
options.app_package = "com.bjjw.cjgc"
|
||||||
|
options.app_activity = ".activity.LoginActivity"
|
||||||
|
|
||||||
|
options.automation_name = "UiAutomator2"
|
||||||
|
options.no_reset = True
|
||||||
|
options.auto_grant_permissions = True
|
||||||
|
options.new_command_timeout = 28800
|
||||||
|
|
||||||
|
# 超时增强(无线 ADB 必须)
|
||||||
|
options.set_capability("uiautomator2ServerLaunchTimeout", 60000)
|
||||||
|
options.set_capability("adbExecTimeout", 120000)
|
||||||
|
|
||||||
|
driver = webdriver.Remote(
|
||||||
|
"http://127.0.0.1:4723",
|
||||||
|
options=options
|
||||||
|
)
|
||||||
|
|
||||||
|
# 使用ADB命令启动Activity
|
||||||
|
try:
|
||||||
|
adb_command = f"adb -s {device_id} shell am start -n com.bjjw.cjgc/.activity.LoginActivity"
|
||||||
|
result = subprocess.run(adb_command, shell=True, capture_output=True, text=True)
|
||||||
|
if result.returncode == 0:
|
||||||
|
time.sleep(1) # 等待Activity启动
|
||||||
|
except Exception:
|
||||||
|
return False
|
||||||
|
|
||||||
|
print("✅ 沉降观测 App 已成功启动")
|
||||||
|
|
||||||
|
# =======================
|
||||||
|
# 获取用户名文本框内容
|
||||||
|
# =======================
|
||||||
|
app_username = None
|
||||||
|
try:
|
||||||
|
print("🔍 尝试获取用户名文本框内容...")
|
||||||
|
|
||||||
|
# 创建显式等待对象
|
||||||
|
wait = WebDriverWait(driver, 15) # 最多等待15秒
|
||||||
|
|
||||||
|
# 等待用户名文本框可点击
|
||||||
|
username_field = wait.until(
|
||||||
|
EC.element_to_be_clickable((AppiumBy.ID, "com.bjjw.cjgc:id/et_user_name"))
|
||||||
|
)
|
||||||
|
|
||||||
|
# 获取文本框中的文本内容
|
||||||
|
app_username = username_field.text
|
||||||
|
|
||||||
|
# 如果文本框是空的,尝试使用get_attribute获取文本
|
||||||
|
if not app_username:
|
||||||
|
app_username = username_field.get_attribute("text")
|
||||||
|
|
||||||
|
# 如果还是空的,尝试获取其他属性
|
||||||
|
if not app_username:
|
||||||
|
app_username = username_field.get_attribute("content-desc") or username_field.get_attribute("label")
|
||||||
|
|
||||||
|
if app_username:
|
||||||
|
print(f"✅ 成功获取到用户名: {app_username}")
|
||||||
|
else:
|
||||||
|
print("⚠️ 用户名文本框为空")
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
print(f"❌ 获取用户名失败: {e}")
|
||||||
|
|
||||||
|
return driver, app_username
|
||||||
|
|
||||||
|
|
||||||
|
# =======================
|
||||||
|
# 无线 ADB 建链主流程
|
||||||
|
# =======================
|
||||||
|
|
||||||
|
def setup_adb_wireless(yh_id="68c0dbfdb7cbcd616e7c5ab5"):
|
||||||
|
port = get_new_port(yh_id)
|
||||||
|
# port = 3435
|
||||||
|
print(f"🚀 开始无线 ADB 建链(端口 {port})")
|
||||||
|
print(f"📋 用户ID: {yh_id}")
|
||||||
|
|
||||||
|
# 从服务器获取账户信息
|
||||||
|
accounts = get_accounts_from_server(yh_id)
|
||||||
|
if not accounts:
|
||||||
|
print("❌ 未从服务器获取到账户信息,终止流程")
|
||||||
|
return
|
||||||
|
|
||||||
|
devices_output = run_command("adb devices")
|
||||||
|
lines = devices_output.splitlines()[1:]
|
||||||
|
|
||||||
|
usb_devices = []
|
||||||
|
|
||||||
|
for line in lines:
|
||||||
|
if not line.strip():
|
||||||
|
continue
|
||||||
|
|
||||||
|
device_id = line.split()[0]
|
||||||
|
|
||||||
|
# 跳过已经是无线的
|
||||||
|
if ":" in device_id:
|
||||||
|
continue
|
||||||
|
|
||||||
|
usb_devices.append(device_id)
|
||||||
|
|
||||||
|
if not usb_devices:
|
||||||
|
print("❌ 未检测到 USB 设备")
|
||||||
|
return
|
||||||
|
|
||||||
|
for serial in usb_devices:
|
||||||
|
print(f"\n🔎 处理设备: {serial}")
|
||||||
|
|
||||||
|
# 获取 WLAN IP
|
||||||
|
ip_info = run_command(f"adb -s {serial} shell ip addr show wlan0")
|
||||||
|
ip_match = re.search(r'inet\s+(\d+\.\d+\.\d+\.\d+)', ip_info)
|
||||||
|
|
||||||
|
if not ip_match:
|
||||||
|
print("⚠️ 获取 IP 失败,请确认已连接 WiFi")
|
||||||
|
continue
|
||||||
|
|
||||||
|
device_ip = ip_match.group(1)
|
||||||
|
print(f"📍 设备 IP: {device_ip}")
|
||||||
|
|
||||||
|
# 切 TCP 模式
|
||||||
|
run_command(f"adb -s {serial} tcpip {port}")
|
||||||
|
time.sleep(2)
|
||||||
|
|
||||||
|
# 无线连接
|
||||||
|
connect_result = run_command(f"adb connect {device_ip}:{port}")
|
||||||
|
|
||||||
|
if "connected" not in connect_result.lower():
|
||||||
|
print(f"❌ 无线连接失败: {connect_result}")
|
||||||
|
continue
|
||||||
|
|
||||||
|
wireless_id = f"{device_ip}:{port}"
|
||||||
|
print(f"✅ 无线 ADB 成功: {wireless_id}")
|
||||||
|
|
||||||
|
# ===== 后续自动化 =====
|
||||||
|
start_appium()
|
||||||
|
driver, app_username = start_settlement_app(wireless_id, device_ip, port)
|
||||||
|
|
||||||
|
if not app_username:
|
||||||
|
print("⚠️ 未获取到App中的用户名,跳过服务器更新")
|
||||||
|
continue
|
||||||
|
|
||||||
|
# 在账户列表中查找匹配的用户名
|
||||||
|
matched_account = None
|
||||||
|
for account in accounts:
|
||||||
|
if account.get("username") == app_username:
|
||||||
|
matched_account = account
|
||||||
|
break
|
||||||
|
|
||||||
|
if not matched_account:
|
||||||
|
print(f"❌ 未找到与用户名 '{app_username}' 匹配的账户")
|
||||||
|
continue
|
||||||
|
|
||||||
|
print(f"✅ 找到匹配账户: {matched_account.get('cl_name')} ({matched_account.get('username')})")
|
||||||
|
print(f" account_id: {matched_account.get('account_id')}")
|
||||||
|
|
||||||
|
# 更新设备信息到服务器
|
||||||
|
device_name = serial # 使用设备序列号作为设备名称
|
||||||
|
|
||||||
|
# 构建更新数据
|
||||||
|
update_data = {
|
||||||
|
"account_id": matched_account.get("account_id"),
|
||||||
|
"device_name": device_name,
|
||||||
|
"device_port": port,
|
||||||
|
"device_ip": device_ip
|
||||||
|
}
|
||||||
|
|
||||||
|
success = update_device_info(
|
||||||
|
account_id=matched_account.get("account_id"),
|
||||||
|
device_name=device_name,
|
||||||
|
device_port=port,
|
||||||
|
device_ip=device_ip
|
||||||
|
)
|
||||||
|
|
||||||
|
if success:
|
||||||
|
print(f"🎉 所有操作完成! 账户 {matched_account.get('username')} 的设备信息已更新")
|
||||||
|
else:
|
||||||
|
print(f"⚠️ 设备信息更新失败,但无线连接和App启动已完成")
|
||||||
|
|
||||||
|
# 关闭Appium连接
|
||||||
|
if driver:
|
||||||
|
print("🔄 关闭Appium连接...")
|
||||||
|
driver.quit()
|
||||||
|
|
||||||
|
break # 处理完第一个设备后退出,如需处理多个设备可移除此行
|
||||||
|
|
||||||
|
|
||||||
|
# =======================
|
||||||
|
# 程序入口
|
||||||
|
# =======================
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
# 配置参数
|
||||||
|
|
||||||
|
USER_ID = "68c0dbfdb7cbcd616e7c5ab5" # 替换为实际的用户ID
|
||||||
|
|
||||||
|
setup_adb_wireless(USER_ID)
|
||||||
38
create_a_link.spec
Normal file
38
create_a_link.spec
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
# -*- mode: python ; coding: utf-8 -*-
|
||||||
|
|
||||||
|
|
||||||
|
a = Analysis(
|
||||||
|
['create_a_link.py'],
|
||||||
|
pathex=[],
|
||||||
|
binaries=[],
|
||||||
|
datas=[],
|
||||||
|
hiddenimports=[],
|
||||||
|
hookspath=[],
|
||||||
|
hooksconfig={},
|
||||||
|
runtime_hooks=[],
|
||||||
|
excludes=[],
|
||||||
|
noarchive=False,
|
||||||
|
optimize=0,
|
||||||
|
)
|
||||||
|
pyz = PYZ(a.pure)
|
||||||
|
|
||||||
|
exe = EXE(
|
||||||
|
pyz,
|
||||||
|
a.scripts,
|
||||||
|
a.binaries,
|
||||||
|
a.datas,
|
||||||
|
[],
|
||||||
|
name='create_a_link',
|
||||||
|
debug=False,
|
||||||
|
bootloader_ignore_signals=False,
|
||||||
|
strip=False,
|
||||||
|
upx=True,
|
||||||
|
upx_exclude=[],
|
||||||
|
runtime_tmpdir=None,
|
||||||
|
console=True,
|
||||||
|
disable_windowed_traceback=False,
|
||||||
|
argv_emulation=False,
|
||||||
|
target_arch=None,
|
||||||
|
codesign_identity=None,
|
||||||
|
entitlements_file=None,
|
||||||
|
)
|
||||||
BIN
dist/create_a_link.exe
vendored
Normal file
BIN
dist/create_a_link.exe
vendored
Normal file
Binary file not shown.
BIN
dist/scheduler_two.exe
vendored
Normal file
BIN
dist/scheduler_two.exe
vendored
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -58,7 +58,7 @@ def get_breakpoint_list():
|
|||||||
"""
|
"""
|
||||||
# 请求参数
|
# 请求参数
|
||||||
params = {
|
params = {
|
||||||
'user_name': global_variable.GLOBAL_USERNAME
|
'user_name': global_variable.get_username()
|
||||||
}
|
}
|
||||||
|
|
||||||
# 请求地址
|
# 请求地址
|
||||||
@@ -114,7 +114,7 @@ def get_measurement_task():
|
|||||||
url = "https://engineering.yuxindazhineng.com/index/index/getOne"
|
url = "https://engineering.yuxindazhineng.com/index/index/getOne"
|
||||||
|
|
||||||
# 获取用户名
|
# 获取用户名
|
||||||
user_name = global_variable.GLOBAL_USERNAME
|
user_name = global_variable.get_username()
|
||||||
if not user_name:
|
if not user_name:
|
||||||
logging.error("未设置用户名,无法获取测量任务")
|
logging.error("未设置用户名,无法获取测量任务")
|
||||||
return None
|
return None
|
||||||
@@ -156,8 +156,8 @@ def get_end_with_num():
|
|||||||
url = "https://engineering.yuxindazhineng.com/index/index/getOne3"
|
url = "https://engineering.yuxindazhineng.com/index/index/getOne3"
|
||||||
|
|
||||||
# 获取用户名
|
# 获取用户名
|
||||||
user_name = global_variable.GLOBAL_USERNAME
|
user_name = global_variable.get_username()
|
||||||
line_num = global_variable.GLOBAL_LINE_NUM
|
line_num = global_variable.get_line_num()
|
||||||
if not line_num:
|
if not line_num:
|
||||||
logging.error("未设置线路编码,无法获取测量任务")
|
logging.error("未设置线路编码,无法获取测量任务")
|
||||||
return None
|
return None
|
||||||
@@ -451,17 +451,18 @@ def get_line_info_and_save_global(user_name: str) -> bool:
|
|||||||
# # 存入全局字典:key=line_num,value=line_name
|
# # 存入全局字典:key=line_num,value=line_name
|
||||||
# global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT[line_num] = line_name
|
# global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT[line_num] = line_name
|
||||||
# 存入全局字典:key=line_name,value=line_num
|
# 存入全局字典:key=line_name,value=line_num
|
||||||
global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT[line_name] = line_num
|
global_variable.get_upload_breakpoint_dict()[line_name] = line_num
|
||||||
|
|
||||||
|
print(f"当前全局字典数据上传线路字典数据:{global_variable.get_upload_breakpoint_dict()}")
|
||||||
# 如果line_name不在列表中,则添加
|
# 如果line_name不在列表中,则添加
|
||||||
if line_name not in global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST:
|
if line_name not in global_variable.get_upload_breakpoint_list():
|
||||||
global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST.append(line_name)
|
global_variable.get_upload_breakpoint_list().append(line_name)
|
||||||
|
|
||||||
logging.info(f"找到status=3的线路信息:line_num={line_num}, line_name={line_name}")
|
logging.info(f"找到status=3的线路信息:line_num={line_num}, line_name={line_name}")
|
||||||
found_valid_data = True
|
found_valid_data = True
|
||||||
|
|
||||||
if found_valid_data:
|
if found_valid_data:
|
||||||
logging.info(f"成功提取所有status=3的线路信息,当前全局字典数据:{global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT}")
|
logging.info(f"成功提取所有status=3的线路信息,当前全局字典数据:{global_variable.get_upload_breakpoint_dict()}")
|
||||||
return True
|
return True
|
||||||
else:
|
else:
|
||||||
logging.warning("data列表中未找到任何status=3且字段完整的线路信息")
|
logging.warning("data列表中未找到任何status=3且字段完整的线路信息")
|
||||||
@@ -476,4 +477,37 @@ def get_line_info_and_save_global(user_name: str) -> bool:
|
|||||||
return False
|
return False
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logging.error(f"调用get_name_all接口时发生未知异常:{str(e)}", exc_info=True) # exc_info=True打印异常堆栈,方便排查
|
logging.error(f"调用get_name_all接口时发生未知异常:{str(e)}", exc_info=True) # exc_info=True打印异常堆栈,方便排查
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
def get_accounts_from_server(yh_id):
|
||||||
|
"""从服务器获取账户信息"""
|
||||||
|
url = "http://www.yuxindazhineng.com:3002/api/accounts/get_uplaod_data"
|
||||||
|
headers = {
|
||||||
|
"Content-Type": "application/json"
|
||||||
|
}
|
||||||
|
data = {
|
||||||
|
"yh_id": yh_id
|
||||||
|
}
|
||||||
|
|
||||||
|
try:
|
||||||
|
print(f"🔍 查询服务器账户信息,用户ID: {yh_id}")
|
||||||
|
response = requests.post(url, headers=headers, json=data, timeout=10)
|
||||||
|
|
||||||
|
if response.status_code == 200:
|
||||||
|
result = response.json()
|
||||||
|
if result.get("code") == 0:
|
||||||
|
print(f"✅ 查询成功,找到 {result.get('total', 0)} 个账户")
|
||||||
|
return result.get("data", [])
|
||||||
|
else:
|
||||||
|
print(f"❌ 查询失败: {result.get('message', '未知错误')}")
|
||||||
|
return []
|
||||||
|
else:
|
||||||
|
print(f"❌ 服务器响应错误: {response.status_code}")
|
||||||
|
return []
|
||||||
|
except requests.exceptions.RequestException as e:
|
||||||
|
print(f"❌ 网络请求失败: {e}")
|
||||||
|
return []
|
||||||
|
except json.JSONDecodeError as e:
|
||||||
|
print(f"❌ JSON解析失败: {e}")
|
||||||
|
return []
|
||||||
|
|
||||||
|
|||||||
@@ -48,10 +48,37 @@ def init_appium_driver(device_id, app_package="com.bjjw.cjgc", app_activity=".ac
|
|||||||
|
|
||||||
try:
|
try:
|
||||||
# 连接Appium服务器
|
# 连接Appium服务器
|
||||||
driver_url = "http://127.0.0.1:4723/wd/hub"
|
# driver_url = "http://127.0.0.1:4723/wd/hub"
|
||||||
logging.info(f"设备 {device_id} 正在连接Appium服务器: {driver_url}")
|
# logging.info(f"设备 {device_id} 正在连接Appium服务器: {driver_url}")
|
||||||
driver = webdriver.Remote(driver_url, options=options)
|
# driver = webdriver.Remote(driver_url, options=options)
|
||||||
logging.info(f"设备 {device_id} Appium服务器连接成功")
|
# logging.info(f"设备 {device_id} Appium服务器连接成功")
|
||||||
|
driver_urls = [
|
||||||
|
"http://127.0.0.1:4723/wd/hub", # 标准路径
|
||||||
|
"http://127.0.0.1:4723", # 简化路径
|
||||||
|
"http://localhost:4723/wd/hub", # localhost
|
||||||
|
]
|
||||||
|
|
||||||
|
driver = None
|
||||||
|
last_exception = None
|
||||||
|
|
||||||
|
# 尝试多个URL
|
||||||
|
for driver_url in driver_urls:
|
||||||
|
try:
|
||||||
|
logging.info(f"设备 {device_id} 正在连接Appium服务器: {driver_url}")
|
||||||
|
driver = webdriver.Remote(driver_url, options=options)
|
||||||
|
logging.info(f"设备 {device_id} Appium服务器连接成功: {driver_url}")
|
||||||
|
break # 连接成功,跳出循环
|
||||||
|
except Exception as e:
|
||||||
|
last_exception = e
|
||||||
|
logging.warning(f"设备 {device_id} 连接失败 {driver_url}: {str(e)[:100]}")
|
||||||
|
continue
|
||||||
|
|
||||||
|
# 检查是否连接成功
|
||||||
|
if not driver:
|
||||||
|
logging.error(f"设备 {device_id} 所有Appium服务器地址尝试失败")
|
||||||
|
logging.error(f"最后错误: {str(last_exception)}")
|
||||||
|
raise Exception(f"设备 {device_id} 无法连接到Appium服务器: {str(last_exception)}")
|
||||||
|
|
||||||
|
|
||||||
# 初始化等待对象
|
# 初始化等待对象
|
||||||
wait = WebDriverWait(driver, 20)
|
wait = WebDriverWait(driver, 20)
|
||||||
@@ -60,6 +87,21 @@ def init_appium_driver(device_id, app_package="com.bjjw.cjgc", app_activity=".ac
|
|||||||
# 等待应用稳定
|
# 等待应用稳定
|
||||||
time.sleep(2)
|
time.sleep(2)
|
||||||
|
|
||||||
|
# 设置屏幕永不休眠
|
||||||
|
try:
|
||||||
|
# 使用ADB命令设置屏幕永不休眠
|
||||||
|
screen_timeout_cmd = [
|
||||||
|
"adb", "-s", device_id,
|
||||||
|
"shell", "settings", "put", "system", "screen_off_timeout", "86400000"
|
||||||
|
]
|
||||||
|
timeout_result = subprocess.run(screen_timeout_cmd, capture_output=True, text=True, timeout=15)
|
||||||
|
if timeout_result.returncode == 0:
|
||||||
|
logging.info(f"设备 {device_id} 已成功设置屏幕永不休眠")
|
||||||
|
else:
|
||||||
|
logging.warning(f"设备 {device_id} 设置屏幕永不休眠失败: {timeout_result.stderr}")
|
||||||
|
except Exception as timeout_error:
|
||||||
|
logging.warning(f"设备 {device_id} 设置屏幕永不休眠时出错: {str(timeout_error)}")
|
||||||
|
|
||||||
logging.info(f"设备 {device_id} Appium驱动初始化完成")
|
logging.info(f"设备 {device_id} Appium驱动初始化完成")
|
||||||
return driver, wait
|
return driver, wait
|
||||||
|
|
||||||
@@ -87,7 +129,7 @@ def check_session_valid(driver, device_id=None):
|
|||||||
bool: 会话有效返回True,否则返回False
|
bool: 会话有效返回True,否则返回False
|
||||||
"""
|
"""
|
||||||
if device_id is None:
|
if device_id is None:
|
||||||
device_id = global_variable.GLOBAL_DEVICE_ID
|
device_id = global_variable.get_device_id()
|
||||||
device_str = f"设备 {device_id} " if device_id else ""
|
device_str = f"设备 {device_id} " if device_id else ""
|
||||||
|
|
||||||
if not driver:
|
if not driver:
|
||||||
@@ -158,7 +200,7 @@ def reconnect_driver(device_id, old_driver=None, app_package="com.bjjw.cjgc", ap
|
|||||||
"""
|
"""
|
||||||
# 使用传入的device_id或从全局变量获取
|
# 使用传入的device_id或从全局变量获取
|
||||||
if not device_id:
|
if not device_id:
|
||||||
device_id = global_variable.GLOBAL_DEVICE_ID
|
device_id = global_variable.get_device_id()
|
||||||
|
|
||||||
# 修复device_id参数类型问题并使用全局设备ID作为备用
|
# 修复device_id参数类型问题并使用全局设备ID作为备用
|
||||||
actual_device_id = device_id
|
actual_device_id = device_id
|
||||||
@@ -174,7 +216,7 @@ def reconnect_driver(device_id, old_driver=None, app_package="com.bjjw.cjgc", ap
|
|||||||
|
|
||||||
# 如果仍然没有有效的设备ID,使用全局变量
|
# 如果仍然没有有效的设备ID,使用全局变量
|
||||||
if not actual_device_id or (isinstance(actual_device_id, str) and ("session=" in actual_device_id or len(actual_device_id.strip()) == 0)):
|
if not actual_device_id or (isinstance(actual_device_id, str) and ("session=" in actual_device_id or len(actual_device_id.strip()) == 0)):
|
||||||
actual_device_id = global_variable.GLOBAL_DEVICE_ID
|
actual_device_id = global_variable.get_device_id()
|
||||||
logging.warning(f"无法获取有效设备ID,使用全局变量GLOBAL_DEVICE_ID: {actual_device_id}")
|
logging.warning(f"无法获取有效设备ID,使用全局变量GLOBAL_DEVICE_ID: {actual_device_id}")
|
||||||
|
|
||||||
device_id = actual_device_id # 使用修正后的设备ID
|
device_id = actual_device_id # 使用修正后的设备ID
|
||||||
@@ -395,7 +437,6 @@ def ensure_appium_server_running(port=4723):
|
|||||||
logging.error("Appium 服务启动超时!请检查 appium 命令是否在命令行可直接运行。")
|
logging.error("Appium 服务启动超时!请检查 appium 命令是否在命令行可直接运行。")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
# ... (保留 init_appium_driver, safe_quit_driver 等其他函数) ...
|
|
||||||
|
|
||||||
def wait_for_appium_start(port, timeout=10):
|
def wait_for_appium_start(port, timeout=10):
|
||||||
"""
|
"""
|
||||||
@@ -442,7 +483,7 @@ def safe_quit_driver(driver, device_id=None):
|
|||||||
device_id: 设备ID(可选)
|
device_id: 设备ID(可选)
|
||||||
"""
|
"""
|
||||||
if device_id is None:
|
if device_id is None:
|
||||||
device_id = global_variable.GLOBAL_DEVICE_ID
|
device_id = global_variable.get_device_id()
|
||||||
device_str = f"设备 {device_id} " if device_id else ""
|
device_str = f"设备 {device_id} " if device_id else ""
|
||||||
logging.info(f"{device_str}开始关闭驱动")
|
logging.info(f"{device_str}开始关闭驱动")
|
||||||
|
|
||||||
@@ -504,7 +545,7 @@ def launch_app_manually(driver, device_id, package_name="com.bjjw.cjgc", activit
|
|||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
if not device_id:
|
if not device_id:
|
||||||
device_id = global_variable.GLOBAL_DEVICE_ID
|
device_id = global_variable.get_device_id()
|
||||||
# 尝试从driver获取设备ID
|
# 尝试从driver获取设备ID
|
||||||
if driver and hasattr(driver, 'capabilities'):
|
if driver and hasattr(driver, 'capabilities'):
|
||||||
device_id = driver.capabilities.get('udid')
|
device_id = driver.capabilities.get('udid')
|
||||||
@@ -665,6 +706,7 @@ def go_main_click_tabber_button(driver, device_id, tabber_button_text, max_retri
|
|||||||
tabber_button = driver.find_element(AppiumBy.ID, tabber_button_text)
|
tabber_button = driver.find_element(AppiumBy.ID, tabber_button_text)
|
||||||
# 点击按钮
|
# 点击按钮
|
||||||
tabber_button.click()
|
tabber_button.click()
|
||||||
|
tabber_button.click()
|
||||||
logging.info(f"设备 {device_id} 已成功点击导航菜单按钮: {tabber_button_text}")
|
logging.info(f"设备 {device_id} 已成功点击导航菜单按钮: {tabber_button_text}")
|
||||||
|
|
||||||
# 等待页面加载
|
# 等待页面加载
|
||||||
|
|||||||
@@ -1,16 +1,147 @@
|
|||||||
# 全局变量
|
# 全局变量
|
||||||
GLOBAL_DEVICE_ID = "" # 设备ID
|
import threading
|
||||||
GLOBAL_USERNAME = "czyuzongwen" # 用户名
|
|
||||||
GLOBAL_CURRENT_PROJECT_NAME = "" # 当前测试项目名称
|
# 线程本地存储,为每个线程创建独立的变量副本
|
||||||
GLOBAL_LINE_NUM = "" # 线路编码
|
thread_local = threading.local()
|
||||||
GLOBAL_BREAKPOINT_STATUS_CODES = [0,3] # 要获取的断点状态码列表
|
|
||||||
GLOBAL_UPLOAD_BREAKPOINT_LIST = []
|
# 共享数据的锁
|
||||||
GLOBAL_UPLOAD_BREAKPOINT_DICT = {}
|
upload_breakpoint_lock = threading.RLock()
|
||||||
GLOBAL_TESTED_BREAKPOINT_LIST = [] # 测量结束的断点列表
|
upload_success_lock = threading.RLock()
|
||||||
LINE_TIME_MAPPING_DICT = {} # 存储所有线路编码和对应的时间的全局字典
|
line_time_mapping_lock = threading.RLock()
|
||||||
GLOBAL_BREAKPOINT_DICT = {} # 存储测量结束的断点名称和对应的线路编码的全局字典
|
breakpoint_dict_lock = threading.RLock()
|
||||||
GLOBAL_NAME_TO_ID_MAP = {} # 存储所有数据员姓名和对应的身份证号的全局字典
|
name_to_id_map_lock = threading.RLock()
|
||||||
GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST = [] # 上传成功的`断点列表
|
|
||||||
|
# 线程安全的全局变量访问函数
|
||||||
|
def get_device_id():
|
||||||
|
"""获取当前线程的设备ID"""
|
||||||
|
if not hasattr(thread_local, 'GLOBAL_DEVICE_ID'):
|
||||||
|
thread_local.GLOBAL_DEVICE_ID = ""
|
||||||
|
return thread_local.GLOBAL_DEVICE_ID
|
||||||
|
|
||||||
|
def set_device_id(device_id):
|
||||||
|
"""设置当前线程的设备ID"""
|
||||||
|
thread_local.GLOBAL_DEVICE_ID = device_id
|
||||||
|
|
||||||
|
def get_username():
|
||||||
|
"""获取用户名(全局共享)"""
|
||||||
|
if not hasattr(thread_local, 'GLOBAL_USERNAME'):
|
||||||
|
thread_local.GLOBAL_USERNAME = "czyuzongwen"
|
||||||
|
return thread_local.GLOBAL_USERNAME
|
||||||
|
|
||||||
|
def set_username(username):
|
||||||
|
"""设置用户名"""
|
||||||
|
thread_local.GLOBAL_USERNAME = username
|
||||||
|
|
||||||
|
def get_current_project_name():
|
||||||
|
"""获取当前测试项目名称"""
|
||||||
|
if not hasattr(thread_local, 'GLOBAL_CURRENT_PROJECT_NAME'):
|
||||||
|
thread_local.GLOBAL_CURRENT_PROJECT_NAME = ""
|
||||||
|
return thread_local.GLOBAL_CURRENT_PROJECT_NAME
|
||||||
|
|
||||||
|
def set_current_project_name(project_name):
|
||||||
|
"""设置当前测试项目名称"""
|
||||||
|
thread_local.GLOBAL_CURRENT_PROJECT_NAME = project_name
|
||||||
|
|
||||||
|
def get_line_num():
|
||||||
|
"""获取线路编码"""
|
||||||
|
if not hasattr(thread_local, 'GLOBAL_LINE_NUM'):
|
||||||
|
thread_local.GLOBAL_LINE_NUM = ""
|
||||||
|
return thread_local.GLOBAL_LINE_NUM
|
||||||
|
|
||||||
|
def set_line_num(line_num):
|
||||||
|
"""设置线路编码"""
|
||||||
|
thread_local.GLOBAL_LINE_NUM = line_num
|
||||||
|
|
||||||
|
def get_breakpoint_status_codes():
|
||||||
|
"""获取要获取的断点状态码列表"""
|
||||||
|
return [0, 3] # 固定值,不需要线程本地存储
|
||||||
|
|
||||||
|
def get_upload_breakpoint_list():
|
||||||
|
"""获取上传断点列表"""
|
||||||
|
if not hasattr(thread_local, 'GLOBAL_UPLOAD_BREAKPOINT_LIST'):
|
||||||
|
thread_local.GLOBAL_UPLOAD_BREAKPOINT_LIST = []
|
||||||
|
return thread_local.GLOBAL_UPLOAD_BREAKPOINT_LIST
|
||||||
|
|
||||||
|
def set_upload_breakpoint_list(breakpoint_list):
|
||||||
|
"""设置上传断点列表"""
|
||||||
|
thread_local.GLOBAL_UPLOAD_BREAKPOINT_LIST = breakpoint_list
|
||||||
|
|
||||||
|
def get_upload_breakpoint_dict():
|
||||||
|
"""获取上传断点字典"""
|
||||||
|
if not hasattr(thread_local, 'GLOBAL_UPLOAD_BREAKPOINT_DICT'):
|
||||||
|
thread_local.GLOBAL_UPLOAD_BREAKPOINT_DICT = {}
|
||||||
|
return thread_local.GLOBAL_UPLOAD_BREAKPOINT_DICT
|
||||||
|
|
||||||
|
def set_upload_breakpoint_dict(breakpoint_dict):
|
||||||
|
"""设置上传断点字典"""
|
||||||
|
thread_local.GLOBAL_UPLOAD_BREAKPOINT_DICT = breakpoint_dict
|
||||||
|
|
||||||
|
def get_tested_breakpoint_list():
|
||||||
|
"""获取测量结束的断点列表"""
|
||||||
|
if not hasattr(thread_local, 'GLOBAL_TESTED_BREAKPOINT_LIST'):
|
||||||
|
thread_local.GLOBAL_TESTED_BREAKPOINT_LIST = []
|
||||||
|
return thread_local.GLOBAL_TESTED_BREAKPOINT_LIST
|
||||||
|
|
||||||
|
def set_tested_breakpoint_list(tested_list):
|
||||||
|
"""设置测量结束的断点列表"""
|
||||||
|
thread_local.GLOBAL_TESTED_BREAKPOINT_LIST = tested_list
|
||||||
|
|
||||||
|
def get_line_time_mapping_dict():
|
||||||
|
"""获取线路编码和对应的时间的字典"""
|
||||||
|
if not hasattr(thread_local, 'LINE_TIME_MAPPING_DICT'):
|
||||||
|
thread_local.LINE_TIME_MAPPING_DICT = {}
|
||||||
|
return thread_local.LINE_TIME_MAPPING_DICT
|
||||||
|
|
||||||
|
def set_line_time_mapping_dict(mapping_dict):
|
||||||
|
"""设置线路编码和对应的时间的字典"""
|
||||||
|
thread_local.LINE_TIME_MAPPING_DICT = mapping_dict
|
||||||
|
|
||||||
|
def get_breakpoint_dict():
|
||||||
|
"""获取测量结束的断点名称和对应的线路编码的字典"""
|
||||||
|
if not hasattr(thread_local, 'GLOBAL_BREAKPOINT_DICT'):
|
||||||
|
thread_local.GLOBAL_BREAKPOINT_DICT = {}
|
||||||
|
return thread_local.GLOBAL_BREAKPOINT_DICT
|
||||||
|
|
||||||
|
def set_breakpoint_dict(breakpoint_dict):
|
||||||
|
"""设置测量结束的断点名称和对应的线路编码的字典"""
|
||||||
|
thread_local.GLOBAL_BREAKPOINT_DICT = breakpoint_dict
|
||||||
|
|
||||||
|
def get_name_to_id_map():
|
||||||
|
"""获取数据员姓名和对应的身份证号的字典"""
|
||||||
|
if not hasattr(thread_local, 'GLOBAL_NAME_TO_ID_MAP'):
|
||||||
|
thread_local.GLOBAL_NAME_TO_ID_MAP = {}
|
||||||
|
return thread_local.GLOBAL_NAME_TO_ID_MAP
|
||||||
|
|
||||||
|
def set_name_to_id_map(name_id_map):
|
||||||
|
"""设置数据员姓名和对应的身份证号的字典"""
|
||||||
|
thread_local.GLOBAL_NAME_TO_ID_MAP = name_id_map
|
||||||
|
|
||||||
|
def get_upload_success_breakpoint_list():
|
||||||
|
"""获取上传成功的断点列表"""
|
||||||
|
if not hasattr(thread_local, 'GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST'):
|
||||||
|
thread_local.GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST = []
|
||||||
|
return thread_local.GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST
|
||||||
|
|
||||||
|
def set_upload_success_breakpoint_list(success_list):
|
||||||
|
"""设置上传成功的断点列表"""
|
||||||
|
thread_local.GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST = success_list
|
||||||
|
|
||||||
|
# 为了保持 ,保留原有的全局变量名称
|
||||||
|
# 但这些将不再被直接使用,而是通过上面的函数访问
|
||||||
|
GLOBAL_DEVICE_ID = "" # 设备ID
|
||||||
|
GLOBAL_USERNAME = "czyuzongwen" # 用户名
|
||||||
|
GLOBAL_CURRENT_PROJECT_NAME = "" # 当前测试项目名称
|
||||||
|
GLOBAL_LINE_NUM = "" # 线路编码
|
||||||
|
GLOBAL_BREAKPOINT_STATUS_CODES = [0,3] # 要获取的断点状态码列表
|
||||||
|
GLOBAL_UPLOAD_BREAKPOINT_LIST = [] #
|
||||||
|
GLOBAL_UPLOAD_BREAKPOINT_DICT = {} #
|
||||||
|
GLOBAL_TESTED_BREAKPOINT_LIST = [] # 测量结束的断点列表
|
||||||
|
LINE_TIME_MAPPING_DICT = {} # 存储所有线路编码和对应的时间的全局字典
|
||||||
|
GLOBAL_BREAKPOINT_DICT = {} # 存储测量结束的断点名称和对应的线路编码的全局字典
|
||||||
|
GLOBAL_NAME_TO_ID_MAP = {} # 存储所有数据员姓名和对应的身份证号的全局字典
|
||||||
|
GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST = [] # 上传成功的`断点列表
|
||||||
|
|
||||||
|
GLOBAL_YU_ID = "68e764f0f9548871c22f93b8" # 宇恒一号ID
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
204
main.py
204
main.py
@@ -32,156 +32,11 @@ logging.basicConfig(
|
|||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
|
||||||
class DeviceAutomation:
|
class DeviceAutomation(object):
|
||||||
@staticmethod
|
|
||||||
def get_device_id() -> str:
|
|
||||||
|
|
||||||
# """
|
|
||||||
# 获取设备ID,优先使用已连接设备,否则使用全局配置
|
|
||||||
# """
|
|
||||||
# try:
|
|
||||||
# # 检查已连接设备
|
|
||||||
# result = subprocess.run(
|
|
||||||
# ["adb", "devices"],
|
|
||||||
# capture_output=True,
|
|
||||||
# text=True,
|
|
||||||
# timeout=10
|
|
||||||
# )
|
|
||||||
|
|
||||||
# # 解析设备列表
|
|
||||||
# for line in result.stdout.strip().split('\n')[1:]:
|
|
||||||
# if line.strip() and "device" in line and "offline" not in line:
|
|
||||||
# device_id = line.split('\t')[0]
|
|
||||||
# logging.info(f"使用已连接设备: {device_id}")
|
|
||||||
# global_variable.GLOBAL_DEVICE_ID = device_id
|
|
||||||
# return device_id
|
|
||||||
|
|
||||||
# except Exception as e:
|
|
||||||
# logging.warning(f"设备检测失败: {e}")
|
|
||||||
|
|
||||||
"""
|
|
||||||
获取设备ID,优先使用无线连接设备,否则尝试开启无线调试,最后使用全局配置
|
|
||||||
"""
|
|
||||||
try:
|
|
||||||
# 检查已连接设备
|
|
||||||
result = subprocess.run(
|
|
||||||
["adb", "devices"],
|
|
||||||
capture_output=True,
|
|
||||||
text=True,
|
|
||||||
timeout=10
|
|
||||||
)
|
|
||||||
|
|
||||||
# 解析设备列表
|
|
||||||
wireless_device_id = None
|
|
||||||
usb_device_id = None
|
|
||||||
|
|
||||||
# 先查找无线连接的设备(IP:端口格式)
|
|
||||||
for line in result.stdout.strip().split('\n')[1:]:
|
|
||||||
if line.strip() and "device" in line and "offline" not in line:
|
|
||||||
current_device = line.split('\t')[0]
|
|
||||||
# 检查是否为IP:端口格式的无线连接
|
|
||||||
if ":" in current_device and any(char.isdigit() for char in current_device):
|
|
||||||
wireless_device_id = current_device
|
|
||||||
logging.info(f"使用无线连接设备: {wireless_device_id}")
|
|
||||||
global_variable.GLOBAL_DEVICE_ID = wireless_device_id
|
|
||||||
return wireless_device_id
|
|
||||||
else:
|
|
||||||
# 记录第一个USB连接的设备
|
|
||||||
if not usb_device_id:
|
|
||||||
usb_device_id = current_device
|
|
||||||
|
|
||||||
# 如果没有找到无线连接的设备,尝试使用USB设备开启无线调试
|
|
||||||
if not wireless_device_id and usb_device_id:
|
|
||||||
logging.info(f"未找到无线连接设备,尝试使用USB设备 {usb_device_id} 开启无线调试")
|
|
||||||
|
|
||||||
# 尝试获取设备IP地址
|
|
||||||
try:
|
|
||||||
import re
|
|
||||||
import time
|
|
||||||
|
|
||||||
ip_result = subprocess.run(
|
|
||||||
["adb", "-s", usb_device_id, "shell", "ip", "-f", "inet", "addr", "show", "wlan0"],
|
|
||||||
capture_output=True,
|
|
||||||
text=True,
|
|
||||||
timeout=10
|
|
||||||
)
|
|
||||||
|
|
||||||
# 解析IP地址
|
|
||||||
ip_output = ip_result.stdout
|
|
||||||
if "inet " in ip_output:
|
|
||||||
# 提取IP地址
|
|
||||||
ip_match = re.search(r'inet\s+(\d+\.\d+\.\d+\.\d+)', ip_output)
|
|
||||||
if ip_match:
|
|
||||||
device_ip = ip_match.group(1)
|
|
||||||
logging.info(f"获取到设备IP地址: {device_ip}")
|
|
||||||
|
|
||||||
# 开启无线调试
|
|
||||||
tcpip_result = subprocess.run(
|
|
||||||
["adb", "-s", usb_device_id, "tcpip", "5555"],
|
|
||||||
capture_output=True,
|
|
||||||
text=True,
|
|
||||||
timeout=10
|
|
||||||
)
|
|
||||||
|
|
||||||
if "restarting in TCP mode port: 5555" in tcpip_result.stdout:
|
|
||||||
logging.info("无线调试已开启,端口: 5555")
|
|
||||||
|
|
||||||
# 等待几秒钟让设备准备好
|
|
||||||
time.sleep(3)
|
|
||||||
|
|
||||||
# 连接到无线设备
|
|
||||||
connect_result = subprocess.run(
|
|
||||||
["adb", "connect", f"{device_ip}:5555"],
|
|
||||||
capture_output=True,
|
|
||||||
text=True,
|
|
||||||
timeout=10
|
|
||||||
)
|
|
||||||
|
|
||||||
if "connected to" in connect_result.stdout:
|
|
||||||
logging.info(f"成功连接到无线设备: {device_ip}:5555")
|
|
||||||
global_variable.GLOBAL_DEVICE_ID = f"{device_ip}:5555"
|
|
||||||
return f"{device_ip}:5555"
|
|
||||||
else:
|
|
||||||
logging.warning(f"连接无线设备失败: {connect_result.stderr}")
|
|
||||||
logging.info(f"使用USB设备: {usb_device_id}")
|
|
||||||
global_variable.GLOBAL_DEVICE_ID = usb_device_id
|
|
||||||
return usb_device_id
|
|
||||||
else:
|
|
||||||
logging.warning(f"开启无线调试失败: {tcpip_result.stderr}")
|
|
||||||
logging.info(f"使用USB设备: {usb_device_id}")
|
|
||||||
global_variable.GLOBAL_DEVICE_ID = usb_device_id
|
|
||||||
return usb_device_id
|
|
||||||
else:
|
|
||||||
logging.warning("未找到设备IP地址")
|
|
||||||
logging.info(f"使用USB设备: {usb_device_id}")
|
|
||||||
global_variable.GLOBAL_DEVICE_ID = usb_device_id
|
|
||||||
return usb_device_id
|
|
||||||
else:
|
|
||||||
logging.warning("无法获取设备IP地址,可能设备未连接到WiFi")
|
|
||||||
logging.info(f"使用USB设备: {usb_device_id}")
|
|
||||||
global_variable.GLOBAL_DEVICE_ID = usb_device_id
|
|
||||||
return usb_device_id
|
|
||||||
except Exception as e:
|
|
||||||
logging.warning(f"开启无线调试时出错: {str(e)}")
|
|
||||||
logging.info(f"使用USB设备: {usb_device_id}")
|
|
||||||
global_variable.GLOBAL_DEVICE_ID = usb_device_id
|
|
||||||
return usb_device_id
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
logging.warning(f"设备检测失败: {e}")
|
|
||||||
|
|
||||||
|
|
||||||
# 使用全局配置
|
|
||||||
device_id = global_variable.GLOBAL_DEVICE_ID
|
|
||||||
logging.info(f"使用全局配置设备: {device_id}")
|
|
||||||
return device_id
|
|
||||||
|
|
||||||
def __init__(self, device_id=None):
|
def __init__(self, device_id=None):
|
||||||
# 如果没有提供设备ID,则自动获取
|
self.device_id = device_id
|
||||||
if device_id is None:
|
|
||||||
self.device_id = self.get_device_id()
|
|
||||||
else:
|
|
||||||
self.device_id = device_id
|
|
||||||
|
|
||||||
# 初始化权限
|
# 初始化权限
|
||||||
if permissions.grant_appium_permissions(self.device_id):
|
if permissions.grant_appium_permissions(self.device_id):
|
||||||
@@ -282,7 +137,7 @@ class DeviceAutomation:
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
# 获取状态为3的线路。
|
# 获取状态为3的线路。
|
||||||
apis.get_line_info_and_save_global(user_name=global_variable.GLOBAL_USERNAME);
|
apis.get_line_info_and_save_global(user_name=global_variable.get_username());
|
||||||
# # 虚拟数据替代
|
# # 虚拟数据替代
|
||||||
# global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT = {'CDWZQ-2标-龙骨湾右线大桥-0-7号墩-平原': 'L156372', 'CDWZQ-2标-蓝家湾特大 桥-31-31-平原': 'L159206'}
|
# global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT = {'CDWZQ-2标-龙骨湾右线大桥-0-7号墩-平原': 'L156372', 'CDWZQ-2标-蓝家湾特大 桥-31-31-平原': 'L159206'}
|
||||||
# global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST = list(global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT.keys())
|
# global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST = list(global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT.keys())
|
||||||
@@ -299,7 +154,7 @@ class DeviceAutomation:
|
|||||||
|
|
||||||
|
|
||||||
# 检查是否有需要上传的断点
|
# 检查是否有需要上传的断点
|
||||||
if not global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST:
|
if not global_variable.get_upload_breakpoint_list():
|
||||||
logging.info(f"设备 {self.device_id} 断点列表为空,无需执行上传操作")
|
logging.info(f"设备 {self.device_id} 断点列表为空,无需执行上传操作")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
@@ -312,12 +167,12 @@ class DeviceAutomation:
|
|||||||
|
|
||||||
# 遍历断点列表,逐个执行上传操作
|
# 遍历断点列表,逐个执行上传操作
|
||||||
upload_success_count = 0
|
upload_success_count = 0
|
||||||
for breakpoint_name in global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST:
|
for breakpoint_name in global_variable.get_upload_breakpoint_list():
|
||||||
try:
|
try:
|
||||||
logging.info(f"设备 {self.device_id} 开始处理断点 '{breakpoint_name}' 的上传")
|
logging.info(f"设备 {self.device_id} 开始处理断点 '{breakpoint_name}' 的上传")
|
||||||
|
|
||||||
# 安全地获取断点信息
|
# 安全地获取断点信息
|
||||||
line_num = global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT.get(breakpoint_name)
|
line_num = global_variable.get_upload_breakpoint_dict().get(breakpoint_name)
|
||||||
if line_num is None:
|
if line_num is None:
|
||||||
logging.warning(f"设备 {self.device_id} 断点 '{breakpoint_name}' 在字典中未找到,跳过上传")
|
logging.warning(f"设备 {self.device_id} 断点 '{breakpoint_name}' 在字典中未找到,跳过上传")
|
||||||
continue
|
continue
|
||||||
@@ -335,25 +190,25 @@ class DeviceAutomation:
|
|||||||
except Exception as e:
|
except Exception as e:
|
||||||
logging.error(f"设备 {self.device_id} 处理断点 '{breakpoint_name}' 时发生异常: {str(e)}")
|
logging.error(f"设备 {self.device_id} 处理断点 '{breakpoint_name}' 时发生异常: {str(e)}")
|
||||||
|
|
||||||
logging.info(f"设备 {self.device_id} 上传配置管理执行完成,成功上传 {upload_success_count}/{len(global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST)} 个断点")
|
logging.info(f"设备 {self.device_id} 上传配置管理执行完成,成功上传 {upload_success_count}/{len(global_variable.get_upload_breakpoint_list())} 个断点")
|
||||||
|
|
||||||
# 如果所有断点都上传成功,返回True;否则返回False
|
# 如果所有断点都上传成功,返回True;否则返回False
|
||||||
all_upload_success = upload_success_count == len(global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST)
|
all_upload_success = upload_success_count == len(global_variable.get_upload_breakpoint_list())
|
||||||
if all_upload_success:
|
if all_upload_success:
|
||||||
logging.info(f"设备 {self.device_id} 所有断点上传成功")
|
logging.info(f"设备 {self.device_id} 所有断点上传成功")
|
||||||
# 把上传成功的断点写入日志文件"上传成功的断点.txt"
|
# 把上传成功的断点写入日志文件"上传成功的断点.txt"
|
||||||
with open(os.path.join(self.results_dir, "上传成功的断点.txt"), "w", encoding='utf-8') as f:
|
with open(os.path.join(self.results_dir, "上传成功的断点.txt"), "w", encoding='utf-8') as f:
|
||||||
for bp in global_variable.GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST:
|
for bp in global_variable.get_upload_success_breakpoint_list():
|
||||||
f.write(f"{bp}\n")
|
f.write(f"{bp}\n")
|
||||||
else:
|
else:
|
||||||
logging.warning(f"设备 {self.device_id} 部分断点上传失败")
|
logging.warning(f"设备 {self.device_id} 部分断点上传失败")
|
||||||
# 把上传成功的断点写入日志文件"上传成功的断点.txt"
|
# 把上传成功的断点写入日志文件"上传成功的断点.txt"
|
||||||
with open(os.path.join(self.results_dir, "上传成功的断点.txt"), "w", encoding='utf-8') as f:
|
with open(os.path.join(self.results_dir, "上传成功的断点.txt"), "w", encoding='utf-8') as f:
|
||||||
for bp in global_variable.GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST:
|
for bp in global_variable.get_upload_success_breakpoint_list():
|
||||||
f.write(f"{bp}\n")
|
f.write(f"{bp}\n")
|
||||||
# 把上传失败的断点写入日志文件"上传失败的断点.txt"
|
# 把上传失败的断点写入日志文件"上传失败的断点.txt"
|
||||||
with open(os.path.join(self.results_dir, "上传失败的断点.txt"), "w", encoding='utf-8') as f:
|
with open(os.path.join(self.results_dir, "上传失败的断点.txt"), "w", encoding='utf-8') as f:
|
||||||
for bp in set(global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST)-set(global_variable.GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST):
|
for bp in set(global_variable.get_upload_breakpoint_list())-set(global_variable.get_upload_success_breakpoint_list()):
|
||||||
f.write(f"{bp}\n")
|
f.write(f"{bp}\n")
|
||||||
|
|
||||||
return all_upload_success
|
return all_upload_success
|
||||||
@@ -457,13 +312,48 @@ class DeviceAutomation:
|
|||||||
"""关闭驱动"""
|
"""关闭驱动"""
|
||||||
safe_quit_driver(getattr(self, 'driver', None), self.device_id)
|
safe_quit_driver(getattr(self, 'driver', None), self.device_id)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def start_upload(device_id=None, upload_time=None):
|
||||||
|
"""
|
||||||
|
供其他页面或模块调用的静态方法
|
||||||
|
执行完整的自动化流程
|
||||||
|
|
||||||
|
参数:
|
||||||
|
device_id: 可选的设备ID,如果为None则自动获取
|
||||||
|
|
||||||
|
返回:
|
||||||
|
bool: 自动化流程执行结果(True/False)
|
||||||
|
"""
|
||||||
|
automation = None
|
||||||
|
try:
|
||||||
|
# 创建自动化实例
|
||||||
|
automation = DeviceAutomation(device_id=device_id)
|
||||||
|
|
||||||
|
# 执行自动化流程
|
||||||
|
success = automation.run_automation()
|
||||||
|
|
||||||
|
if success:
|
||||||
|
logging.info("自动化流程执行成功")
|
||||||
|
else:
|
||||||
|
logging.error("自动化流程执行失败")
|
||||||
|
|
||||||
|
return success
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"自动化流程执行出错: {str(e)}")
|
||||||
|
return False
|
||||||
|
finally:
|
||||||
|
# 确保资源被清理
|
||||||
|
if automation:
|
||||||
|
automation.quit()
|
||||||
|
|
||||||
# 主执行逻辑
|
# 主执行逻辑
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|
||||||
# 单个设备配置 - 现在DeviceAutomation会自动获取设备ID
|
# 单个设备配置 - 现在DeviceAutomation会自动获取设备ID
|
||||||
|
|
||||||
try:
|
try:
|
||||||
automation = DeviceAutomation()
|
automation = DeviceAutomation(device_id="192.168.1.100:5556")
|
||||||
success = automation.run_automation()
|
success = automation.run_automation()
|
||||||
|
|
||||||
if success:
|
if success:
|
||||||
|
|||||||
478
main_init.py
Normal file
478
main_init.py
Normal file
@@ -0,0 +1,478 @@
|
|||||||
|
# actions.py 主自动化脚本
|
||||||
|
import os
|
||||||
|
import logging
|
||||||
|
import time
|
||||||
|
import subprocess
|
||||||
|
from appium import webdriver
|
||||||
|
from appium.options.android import UiAutomator2Options
|
||||||
|
from appium.webdriver.common.appiumby import AppiumBy
|
||||||
|
from selenium.webdriver.support.ui import WebDriverWait
|
||||||
|
from selenium.webdriver.support import expected_conditions as EC
|
||||||
|
from selenium.common.exceptions import TimeoutException, NoSuchElementException
|
||||||
|
|
||||||
|
import globals.ids as ids
|
||||||
|
import globals.global_variable as global_variable # 导入全局变量模块
|
||||||
|
import permissions # 导入权限处理模块
|
||||||
|
import globals.apis as apis
|
||||||
|
from globals.driver_utils import init_appium_driver, ensure_appium_server_running, safe_quit_driver, is_app_launched, launch_app_manually
|
||||||
|
from page_objects.login_page import LoginPage
|
||||||
|
from page_objects.download_tabbar_page import DownloadTabbarPage
|
||||||
|
from page_objects.screenshot_page import ScreenshotPage
|
||||||
|
from page_objects.upload_config_page import UploadConfigPage
|
||||||
|
from page_objects.more_download_page import MoreDownloadPage
|
||||||
|
|
||||||
|
|
||||||
|
# 配置日志
|
||||||
|
logging.basicConfig(
|
||||||
|
level=logging.INFO,
|
||||||
|
format="%(asctime)s - %(levelname)s: %(message)s",
|
||||||
|
handlers=[
|
||||||
|
logging.FileHandler("appium_automation.log"),
|
||||||
|
logging.StreamHandler()
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
|
class DeviceAutomation:
|
||||||
|
@staticmethod
|
||||||
|
def get_device_id() -> str:
|
||||||
|
|
||||||
|
# """
|
||||||
|
# 获取设备ID,优先使用已连接设备,否则使用全局配置
|
||||||
|
# """
|
||||||
|
# try:
|
||||||
|
# # 检查已连接设备
|
||||||
|
# result = subprocess.run(
|
||||||
|
# ["adb", "devices"],
|
||||||
|
# capture_output=True,
|
||||||
|
# text=True,
|
||||||
|
# timeout=10
|
||||||
|
# )
|
||||||
|
|
||||||
|
# # 解析设备列表
|
||||||
|
# for line in result.stdout.strip().split('\n')[1:]:
|
||||||
|
# if line.strip() and "device" in line and "offline" not in line:
|
||||||
|
# device_id = line.split('\t')[0]
|
||||||
|
# logging.info(f"使用已连接设备: {device_id}")
|
||||||
|
# global_variable.GLOBAL_DEVICE_ID = device_id
|
||||||
|
# return device_id
|
||||||
|
|
||||||
|
# except Exception as e:
|
||||||
|
# logging.warning(f"设备检测失败: {e}")
|
||||||
|
|
||||||
|
"""
|
||||||
|
获取设备ID,优先使用无线连接设备,否则尝试开启无线调试,最后使用全局配置
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
# 检查已连接设备
|
||||||
|
result = subprocess.run(
|
||||||
|
["adb", "devices"],
|
||||||
|
capture_output=True,
|
||||||
|
text=True,
|
||||||
|
timeout=10
|
||||||
|
)
|
||||||
|
|
||||||
|
# 解析设备列表
|
||||||
|
wireless_device_id = None
|
||||||
|
usb_device_id = None
|
||||||
|
|
||||||
|
# 先查找无线连接的设备(IP:端口格式)
|
||||||
|
for line in result.stdout.strip().split('\n')[1:]:
|
||||||
|
if line.strip() and "device" in line and "offline" not in line:
|
||||||
|
current_device = line.split('\t')[0]
|
||||||
|
# 检查是否为IP:端口格式的无线连接
|
||||||
|
if ":" in current_device and any(char.isdigit() for char in current_device):
|
||||||
|
wireless_device_id = current_device
|
||||||
|
logging.info(f"使用无线连接设备: {wireless_device_id}")
|
||||||
|
global_variable.GLOBAL_DEVICE_ID = wireless_device_id
|
||||||
|
return wireless_device_id
|
||||||
|
else:
|
||||||
|
# 记录第一个USB连接的设备
|
||||||
|
if not usb_device_id:
|
||||||
|
usb_device_id = current_device
|
||||||
|
|
||||||
|
# 如果没有找到无线连接的设备,尝试使用USB设备开启无线调试
|
||||||
|
if not wireless_device_id and usb_device_id:
|
||||||
|
logging.info(f"未找到无线连接设备,尝试使用USB设备 {usb_device_id} 开启无线调试")
|
||||||
|
|
||||||
|
# 尝试获取设备IP地址
|
||||||
|
try:
|
||||||
|
import re
|
||||||
|
import time
|
||||||
|
|
||||||
|
ip_result = subprocess.run(
|
||||||
|
["adb", "-s", usb_device_id, "shell", "ip", "-f", "inet", "addr", "show", "wlan0"],
|
||||||
|
capture_output=True,
|
||||||
|
text=True,
|
||||||
|
timeout=10
|
||||||
|
)
|
||||||
|
|
||||||
|
# 解析IP地址
|
||||||
|
ip_output = ip_result.stdout
|
||||||
|
if "inet " in ip_output:
|
||||||
|
# 提取IP地址
|
||||||
|
ip_match = re.search(r'inet\s+(\d+\.\d+\.\d+\.\d+)', ip_output)
|
||||||
|
if ip_match:
|
||||||
|
device_ip = ip_match.group(1)
|
||||||
|
logging.info(f"获取到设备IP地址: {device_ip}")
|
||||||
|
|
||||||
|
# 开启无线调试
|
||||||
|
tcpip_result = subprocess.run(
|
||||||
|
["adb", "-s", usb_device_id, "tcpip", "5555"],
|
||||||
|
capture_output=True,
|
||||||
|
text=True,
|
||||||
|
timeout=10
|
||||||
|
)
|
||||||
|
|
||||||
|
if "restarting in TCP mode port: 5555" in tcpip_result.stdout:
|
||||||
|
logging.info("无线调试已开启,端口: 5555")
|
||||||
|
|
||||||
|
# 等待几秒钟让设备准备好
|
||||||
|
time.sleep(3)
|
||||||
|
|
||||||
|
# 连接到无线设备
|
||||||
|
connect_result = subprocess.run(
|
||||||
|
["adb", "connect", f"{device_ip}:5555"],
|
||||||
|
capture_output=True,
|
||||||
|
text=True,
|
||||||
|
timeout=10
|
||||||
|
)
|
||||||
|
|
||||||
|
if "connected to" in connect_result.stdout:
|
||||||
|
logging.info(f"成功连接到无线设备: {device_ip}:5555")
|
||||||
|
global_variable.GLOBAL_DEVICE_ID = f"{device_ip}:5555"
|
||||||
|
return f"{device_ip}:5555"
|
||||||
|
else:
|
||||||
|
logging.warning(f"连接无线设备失败: {connect_result.stderr}")
|
||||||
|
logging.info(f"使用USB设备: {usb_device_id}")
|
||||||
|
global_variable.GLOBAL_DEVICE_ID = usb_device_id
|
||||||
|
return usb_device_id
|
||||||
|
else:
|
||||||
|
logging.warning(f"开启无线调试失败: {tcpip_result.stderr}")
|
||||||
|
logging.info(f"使用USB设备: {usb_device_id}")
|
||||||
|
global_variable.GLOBAL_DEVICE_ID = usb_device_id
|
||||||
|
return usb_device_id
|
||||||
|
else:
|
||||||
|
logging.warning("未找到设备IP地址")
|
||||||
|
logging.info(f"使用USB设备: {usb_device_id}")
|
||||||
|
global_variable.GLOBAL_DEVICE_ID = usb_device_id
|
||||||
|
return usb_device_id
|
||||||
|
else:
|
||||||
|
logging.warning("无法获取设备IP地址,可能设备未连接到WiFi")
|
||||||
|
logging.info(f"使用USB设备: {usb_device_id}")
|
||||||
|
global_variable.GLOBAL_DEVICE_ID = usb_device_id
|
||||||
|
return usb_device_id
|
||||||
|
except Exception as e:
|
||||||
|
logging.warning(f"开启无线调试时出错: {str(e)}")
|
||||||
|
logging.info(f"使用USB设备: {usb_device_id}")
|
||||||
|
global_variable.GLOBAL_DEVICE_ID = usb_device_id
|
||||||
|
return usb_device_id
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
logging.warning(f"设备检测失败: {e}")
|
||||||
|
|
||||||
|
|
||||||
|
# 使用全局配置
|
||||||
|
device_id = global_variable.GLOBAL_DEVICE_ID
|
||||||
|
logging.info(f"使用全局配置设备: {device_id}")
|
||||||
|
return device_id
|
||||||
|
|
||||||
|
def __init__(self, device_id=None):
|
||||||
|
# 如果没有提供设备ID,则自动获取
|
||||||
|
if device_id is None:
|
||||||
|
self.device_id = self.get_device_id()
|
||||||
|
else:
|
||||||
|
self.device_id = device_id
|
||||||
|
|
||||||
|
# 初始化权限
|
||||||
|
if permissions.grant_appium_permissions(self.device_id):
|
||||||
|
logging.info(f"设备 {self.device_id} 权限授予成功")
|
||||||
|
else:
|
||||||
|
logging.warning(f"设备 {self.device_id} 权限授予失败")
|
||||||
|
|
||||||
|
# 确保Appium服务器正在运行
|
||||||
|
ensure_appium_server_running(4723)
|
||||||
|
|
||||||
|
# 初始化驱动
|
||||||
|
self.init_driver()
|
||||||
|
# 先拼接,后创建测试结果目录
|
||||||
|
self.results_dir = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'test_results')
|
||||||
|
os.makedirs(self.results_dir, exist_ok=True)
|
||||||
|
|
||||||
|
|
||||||
|
def init_driver(self):
|
||||||
|
"""初始化Appium驱动"""
|
||||||
|
try:
|
||||||
|
# 使用全局函数初始化驱动
|
||||||
|
self.driver, self.wait = init_appium_driver(self.device_id)
|
||||||
|
# 初始化页面对象
|
||||||
|
logging.info(f"设备 {self.device_id} 开始初始化页面对象")
|
||||||
|
self.login_page = LoginPage(self.driver, self.wait)
|
||||||
|
self.download_tabbar_page = DownloadTabbarPage(self.driver, self.wait, self.device_id)
|
||||||
|
self.screenshot_page = ScreenshotPage(self.driver, self.wait, self.device_id)
|
||||||
|
self.upload_config_page = UploadConfigPage(self.driver, self.wait, self.device_id)
|
||||||
|
self.more_download_page = MoreDownloadPage(self.driver, self.wait,self.device_id)
|
||||||
|
logging.info(f"设备 {self.device_id} 所有页面对象初始化完成")
|
||||||
|
|
||||||
|
# 检查应用是否成功启动
|
||||||
|
if is_app_launched(self.driver):
|
||||||
|
logging.info(f"设备 {self.device_id} 沉降观测App已成功启动")
|
||||||
|
else:
|
||||||
|
logging.warning(f"设备 {self.device_id} 应用可能未正确启动")
|
||||||
|
# 手动启动应用
|
||||||
|
launch_app_manually(self.driver, self.device_id)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"设备 {self.device_id} 初始化驱动失败: {str(e)}")
|
||||||
|
raise
|
||||||
|
|
||||||
|
def is_app_launched(self):
|
||||||
|
"""检查应用是否已启动"""
|
||||||
|
try:
|
||||||
|
return is_app_launched(self.driver)
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"设备 {self.device_id} 检查应用启动状态时出错: {str(e)}")
|
||||||
|
return False
|
||||||
|
|
||||||
|
def is_element_present(self, by, value):
|
||||||
|
"""检查元素是否存在"""
|
||||||
|
try:
|
||||||
|
self.driver.find_element(by, value)
|
||||||
|
return True
|
||||||
|
except NoSuchElementException:
|
||||||
|
return False
|
||||||
|
|
||||||
|
def handle_app_state(self):
|
||||||
|
"""根据当前应用状态处理相应的操作"""
|
||||||
|
try:
|
||||||
|
login_btn_exists = self.login_page.is_login_page()
|
||||||
|
if not login_btn_exists:
|
||||||
|
logging.error(f"设备 {self.device_id} 未知应用状态,无法确定当前页面,跳转到登录页面")
|
||||||
|
if self.navigate_to_login_page(self.driver, self.device_id):
|
||||||
|
logging.info(f"设备 {self.device_id} 成功跳转到登录页面")
|
||||||
|
return self.handle_app_state() # 递归调用处理登录后的状态
|
||||||
|
else:
|
||||||
|
logging.error(f"设备 {self.device_id} 跳转到登录页面失败")
|
||||||
|
return False
|
||||||
|
|
||||||
|
# 处理登录页面状态
|
||||||
|
logging.info(f"设备 {self.device_id} 检测到登录页面,执行登录操作")
|
||||||
|
max_retries = 1
|
||||||
|
login_success = False
|
||||||
|
|
||||||
|
for attempt in range(max_retries + 1):
|
||||||
|
if self.login_page.login():
|
||||||
|
login_success = True
|
||||||
|
break
|
||||||
|
else:
|
||||||
|
if attempt < max_retries:
|
||||||
|
logging.warning(f"设备 {self.device_id} 登录失败,准备重试 ({attempt + 1}/{max_retries})")
|
||||||
|
time.sleep(2) # 等待2秒后重试
|
||||||
|
else:
|
||||||
|
logging.error(f"设备 {self.device_id} 登录失败,已达到最大重试次数")
|
||||||
|
|
||||||
|
if not login_success:
|
||||||
|
return False
|
||||||
|
|
||||||
|
logging.info(f"设备 {self.device_id} 登录成功,继续执行更新操作")
|
||||||
|
time.sleep(1)
|
||||||
|
|
||||||
|
# 执行更新操作
|
||||||
|
if not self.download_tabbar_page.download_tabbar_page_manager():
|
||||||
|
logging.error(f"设备 {self.device_id} 更新操作执行失败")
|
||||||
|
return False
|
||||||
|
|
||||||
|
# 获取状态为3的线路。
|
||||||
|
apis.get_line_info_and_save_global(user_name=global_variable.GLOBAL_USERNAME);
|
||||||
|
# # 虚拟数据替代
|
||||||
|
# global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT = {'CDWZQ-2标-龙骨湾右线大桥-0-7号墩-平原': 'L156372', 'CDWZQ-2标-蓝家湾特大 桥-31-31-平原': 'L159206'}
|
||||||
|
# global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST = list(global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT.keys())
|
||||||
|
|
||||||
|
# 点击测量导航栏按钮
|
||||||
|
measure_page_btn = WebDriverWait(self.driver, 5).until(
|
||||||
|
EC.element_to_be_clickable((AppiumBy.ID, "com.bjjw.cjgc:id/img_3_layout"))
|
||||||
|
)
|
||||||
|
measure_page_btn.click()
|
||||||
|
|
||||||
|
|
||||||
|
# 处理平差
|
||||||
|
self.screenshot_page.screenshot_page_manager(self.device_id)
|
||||||
|
|
||||||
|
|
||||||
|
# 检查是否有需要上传的断点
|
||||||
|
if not global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST:
|
||||||
|
logging.info(f"设备 {self.device_id} 断点列表为空,无需执行上传操作")
|
||||||
|
return False
|
||||||
|
|
||||||
|
# 点击上传导航栏按钮
|
||||||
|
upload_page_btn = WebDriverWait(self.driver, 5).until(
|
||||||
|
EC.element_to_be_clickable((AppiumBy.ID, "com.bjjw.cjgc:id/img_2_layout"))
|
||||||
|
)
|
||||||
|
upload_page_btn.click()
|
||||||
|
|
||||||
|
|
||||||
|
# 遍历断点列表,逐个执行上传操作
|
||||||
|
upload_success_count = 0
|
||||||
|
for breakpoint_name in global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST:
|
||||||
|
try:
|
||||||
|
logging.info(f"设备 {self.device_id} 开始处理断点 '{breakpoint_name}' 的上传")
|
||||||
|
|
||||||
|
# 安全地获取断点信息
|
||||||
|
line_num = global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT.get(breakpoint_name)
|
||||||
|
if line_num is None:
|
||||||
|
logging.warning(f"设备 {self.device_id} 断点 '{breakpoint_name}' 在字典中未找到,跳过上传")
|
||||||
|
continue
|
||||||
|
if not line_num:
|
||||||
|
logging.warning(f"设备 {self.device_id} 断点 '{breakpoint_name}' 未获取到line_num,跳过上传")
|
||||||
|
continue
|
||||||
|
|
||||||
|
# 执行上传配置管理,传入当前断点名称
|
||||||
|
if self.upload_config_page.upload_config_page_manager(self.results_dir, breakpoint_name, line_num):
|
||||||
|
logging.info(f"设备 {self.device_id} 断点 '{breakpoint_name}' 上传成功")
|
||||||
|
upload_success_count += 1
|
||||||
|
else:
|
||||||
|
logging.error(f"设备 {self.device_id} 断点 '{breakpoint_name}' 上传失败")
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"设备 {self.device_id} 处理断点 '{breakpoint_name}' 时发生异常: {str(e)}")
|
||||||
|
|
||||||
|
logging.info(f"设备 {self.device_id} 上传配置管理执行完成,成功上传 {upload_success_count}/{len(global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST)} 个断点")
|
||||||
|
|
||||||
|
# 如果所有断点都上传成功,返回True;否则返回False
|
||||||
|
all_upload_success = upload_success_count == len(global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST)
|
||||||
|
if all_upload_success:
|
||||||
|
logging.info(f"设备 {self.device_id} 所有断点上传成功")
|
||||||
|
# 把上传成功的断点写入日志文件"上传成功的断点.txt"
|
||||||
|
with open(os.path.join(self.results_dir, "上传成功的断点.txt"), "w", encoding='utf-8') as f:
|
||||||
|
for bp in global_variable.GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST:
|
||||||
|
f.write(f"{bp}\n")
|
||||||
|
else:
|
||||||
|
logging.warning(f"设备 {self.device_id} 部分断点上传失败")
|
||||||
|
# 把上传成功的断点写入日志文件"上传成功的断点.txt"
|
||||||
|
with open(os.path.join(self.results_dir, "上传成功的断点.txt"), "w", encoding='utf-8') as f:
|
||||||
|
for bp in global_variable.GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST:
|
||||||
|
f.write(f"{bp}\n")
|
||||||
|
# 把上传失败的断点写入日志文件"上传失败的断点.txt"
|
||||||
|
with open(os.path.join(self.results_dir, "上传失败的断点.txt"), "w", encoding='utf-8') as f:
|
||||||
|
for bp in set(global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST)-set(global_variable.GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST):
|
||||||
|
f.write(f"{bp}\n")
|
||||||
|
|
||||||
|
return all_upload_success
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"设备 {self.device_id} 处理应用状态时出错: {str(e)}")
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def check_and_click_confirm_popup_appium(self):
|
||||||
|
"""
|
||||||
|
适用于Appium的弹窗检测函数
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool: 是否成功处理弹窗
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
from appium.webdriver.common.appiumby import AppiumBy
|
||||||
|
|
||||||
|
# 使用self.driver而不是参数
|
||||||
|
if not hasattr(self, 'driver') or self.driver is None:
|
||||||
|
logging.warning("driver未初始化,无法检测弹窗")
|
||||||
|
return False
|
||||||
|
|
||||||
|
# 检查弹窗消息
|
||||||
|
message_elements = self.driver.find_elements(AppiumBy.XPATH, '//android.widget.TextView[@text="是否退出测量界面?"]')
|
||||||
|
|
||||||
|
if message_elements:
|
||||||
|
logging.info("检测到退出测量界面弹窗")
|
||||||
|
|
||||||
|
# 点击"是"按钮
|
||||||
|
confirm_buttons = self.driver.find_elements(AppiumBy.XPATH, '//android.widget.Button[@text="是" and @resource-id="android:id/button1"]')
|
||||||
|
if confirm_buttons:
|
||||||
|
confirm_buttons[0].click()
|
||||||
|
logging.info("已点击'是'按钮")
|
||||||
|
time.sleep(1)
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
logging.warning("未找到'是'按钮")
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"Appium检测弹窗时发生错误: {str(e)}")
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def navigate_to_login_page(self, driver, device_id):
|
||||||
|
"""
|
||||||
|
补充的跳转页面函数:当设备处于未知状态时,尝试跳转到登录页面
|
||||||
|
|
||||||
|
参数:
|
||||||
|
driver: 已初始化的Appium WebDriver对象
|
||||||
|
device_id: 设备ID,用于日志记录
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
target_package = 'com.bjjw.cjgc'
|
||||||
|
target_activity = '.activity.LoginActivity'
|
||||||
|
# 使用ADB命令启动Activity
|
||||||
|
try:
|
||||||
|
logging.info(f"尝试使用ADB命令启动LoginActivity: {target_package}/{target_activity}")
|
||||||
|
adb_command = f"adb -s {device_id} shell am start -n {target_package}/{target_activity}"
|
||||||
|
result = subprocess.run(adb_command, shell=True, capture_output=True, text=True)
|
||||||
|
if result.returncode == 0:
|
||||||
|
logging.info(f"使用ADB命令启动LoginActivity成功")
|
||||||
|
time.sleep(2) # 等待Activity启动
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
logging.warning(f"ADB命令执行失败: {result.stderr}")
|
||||||
|
except Exception as adb_error:
|
||||||
|
logging.warning(f"执行ADB命令时出错: {adb_error}")
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"跳转到登录页面过程中发生未预期错误: {e}")
|
||||||
|
|
||||||
|
# 所有尝试都失败
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def run_automation(self):
|
||||||
|
"""运行自动化流程"""
|
||||||
|
try:
|
||||||
|
success = self.handle_app_state()
|
||||||
|
# success = self.test_handle_app_state()
|
||||||
|
if success:
|
||||||
|
logging.info(f"设备 {self.device_id} 自动化流程执行成功")
|
||||||
|
else:
|
||||||
|
logging.error(f"设备 {self.device_id} 自动化流程执行失败")
|
||||||
|
return success
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"设备 {self.device_id} 自动化执行过程中发生错误: {str(e)}")
|
||||||
|
return False
|
||||||
|
finally:
|
||||||
|
self.quit()
|
||||||
|
|
||||||
|
def quit(self):
|
||||||
|
"""关闭驱动"""
|
||||||
|
safe_quit_driver(getattr(self, 'driver', None), self.device_id)
|
||||||
|
|
||||||
|
# 主执行逻辑
|
||||||
|
if __name__ == "__main__":
|
||||||
|
|
||||||
|
# 单个设备配置 - 现在DeviceAutomation会自动获取设备ID
|
||||||
|
|
||||||
|
try:
|
||||||
|
automation = DeviceAutomation()
|
||||||
|
success = automation.run_automation()
|
||||||
|
|
||||||
|
if success:
|
||||||
|
logging.info(f"设备自动化流程执行成功")
|
||||||
|
else:
|
||||||
|
logging.error(f"设备自动化流程执行失败")
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"设备执行出错: {str(e)}")
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -295,25 +295,25 @@ class DownloadTabbarPage:
|
|||||||
"""执行基础更新操作"""
|
"""执行基础更新操作"""
|
||||||
try:
|
try:
|
||||||
# 执行基础更新流程
|
# 执行基础更新流程
|
||||||
self.logger.info(f"设备 {global_variable.GLOBAL_DEVICE_ID} 开始执行更新流程")
|
self.logger.info(f"设备 {global_variable.get_device_id()} 开始执行更新流程")
|
||||||
|
|
||||||
# 点击下载标签栏
|
# 点击下载标签栏
|
||||||
if not self.click_download_tabbar():
|
if not self.click_download_tabbar():
|
||||||
self.logger.error(f"设备 {global_variable.GLOBAL_DEVICE_ID} 点击下载标签栏失败")
|
self.logger.error(f"设备 {global_variable.get_device_id()} 点击下载标签栏失败")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
# 更新工作基点
|
# 更新工作基点
|
||||||
if not self.update_work_base():
|
if not self.update_work_base():
|
||||||
self.logger.error(f"设备 {global_variable.GLOBAL_DEVICE_ID} 更新工作基点失败")
|
self.logger.error(f"设备 {global_variable.get_device_id()} 更新工作基点失败")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
# 更新水准线路
|
# 更新水准线路
|
||||||
if not self.update_level_line():
|
if not self.update_level_line():
|
||||||
self.logger.error(f"设备 {global_variable.GLOBAL_DEVICE_ID} 更新水准线路失败")
|
self.logger.error(f"设备 {global_variable.get_device_id()} 更新水准线路失败")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
self.logger.info(f"设备 {global_variable.GLOBAL_DEVICE_ID} 更新操作执行成功")
|
self.logger.info(f"设备 {global_variable.get_device_id()} 更新操作执行成功")
|
||||||
return True
|
return True
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.logger.error(f"设备 {global_variable.GLOBAL_DEVICE_ID} 执行更新操作时出错: {str(e)}")
|
self.logger.error(f"设备 {global_variable.get_device_id()} 执行更新操作时出错: {str(e)}")
|
||||||
return False
|
return False
|
||||||
@@ -6,6 +6,7 @@ from selenium.webdriver.support import expected_conditions as EC
|
|||||||
from selenium.common.exceptions import TimeoutException, NoSuchElementException
|
from selenium.common.exceptions import TimeoutException, NoSuchElementException
|
||||||
import logging
|
import logging
|
||||||
import time
|
import time
|
||||||
|
import globals.apis as apis
|
||||||
|
|
||||||
import globals.ids as ids
|
import globals.ids as ids
|
||||||
import globals.global_variable as global_variable # 导入全局变量模块
|
import globals.global_variable as global_variable # 导入全局变量模块
|
||||||
@@ -35,14 +36,41 @@ class LoginPage:
|
|||||||
|
|
||||||
# 读取文本框内已有的用户名(.text属性获取元素显示的文本内容)
|
# 读取文本框内已有的用户名(.text属性获取元素显示的文本内容)
|
||||||
existing_username = username_field.text
|
existing_username = username_field.text
|
||||||
# 3. 将获取到的用户名写入全局变量中
|
# 3. 将获取到的用户名写入全局变量中
|
||||||
global_variable.GLOBAL_USERNAME = existing_username # 关键:给全局变量赋值
|
# global_variable.GLOBAL_USERNAME = existing_username # 关键:给全局变量赋值
|
||||||
|
global_variable.set_username(existing_username)
|
||||||
|
|
||||||
# 日志记录获取到的已有用户名(若为空,也需明确记录,避免后续误解)
|
# 日志记录获取到的已有用户名(若为空,也需明确记录,避免后续误解)
|
||||||
if existing_username.strip(): # 去除空格后判断是否有有效内容
|
if existing_username.strip(): # 去除空格后判断是否有有效内容
|
||||||
self.logger.info(f"已获取文本框中的已有用户名: {existing_username}")
|
self.logger.info(f"已获取文本框中的已有用户名: {existing_username}")
|
||||||
else:
|
else:
|
||||||
self.logger.info("文本框中未检测到已有用户名(内容为空)")
|
self.logger.info("文本框中未检测到已有用户名(内容为空)")
|
||||||
|
|
||||||
|
# 1. 定位密码输入框
|
||||||
|
password_field = self.wait.until(
|
||||||
|
EC.element_to_be_clickable((AppiumBy.ID, ids.LOGIN_PASSWORD))
|
||||||
|
)
|
||||||
|
|
||||||
|
# 2. 清空密码框(如果需要)
|
||||||
|
try:
|
||||||
|
password_field.clear()
|
||||||
|
# time.sleep(0.5) # 等待清除完成
|
||||||
|
except:
|
||||||
|
# 如果clear方法不可用,尝试其他方式
|
||||||
|
pass
|
||||||
|
|
||||||
|
accounts = apis.get_accounts_from_server("68c0dbfdb7cbcd616e7c5ab5")
|
||||||
|
matches = [acc for acc in accounts if acc.get("username") == existing_username]
|
||||||
|
if matches:
|
||||||
|
password = matches[0].get("password")
|
||||||
|
|
||||||
|
password_field.send_keys(password)
|
||||||
|
|
||||||
|
# 4. 可选:隐藏键盘
|
||||||
|
try:
|
||||||
|
self.driver.hide_keyboard()
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
# 点击登录按钮
|
# 点击登录按钮
|
||||||
login_btn = self.wait.until(
|
login_btn = self.wait.until(
|
||||||
|
|||||||
@@ -258,8 +258,8 @@ class ScreenshotPage:
|
|||||||
tuple: (date_str, time_str) 日期和时间字符串
|
tuple: (date_str, time_str) 日期和时间字符串
|
||||||
"""
|
"""
|
||||||
|
|
||||||
if line_code in global_variable.LINE_TIME_MAPPING_DICT:
|
if line_code in global_variable.get_line_time_mapping_dict():
|
||||||
end_time = global_variable.LINE_TIME_MAPPING_DICT[line_code]
|
end_time = global_variable.get_line_time_mapping_dict()[line_code]
|
||||||
date_str = end_time.strftime("%Y-%m-%d")
|
date_str = end_time.strftime("%Y-%m-%d")
|
||||||
time_str = end_time.strftime("%H:%M:%S")
|
time_str = end_time.strftime("%H:%M:%S")
|
||||||
return (date_str, time_str)
|
return (date_str, time_str)
|
||||||
@@ -273,19 +273,19 @@ class ScreenshotPage:
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
self.logger.info("\n当前全局字典内容:")
|
self.logger.info("\n当前全局字典内容:")
|
||||||
if global_variable.LINE_TIME_MAPPING_DICT:
|
if global_variable.get_line_time_mapping_dict():
|
||||||
for line_code, end_time in sorted(global_variable.LINE_TIME_MAPPING_DICT.items()):
|
for line_code, end_time in sorted(global_variable.get_line_time_mapping_dict().items()):
|
||||||
date_str = end_time.strftime("%Y-%m-%d")
|
date_str = end_time.strftime("%Y-%m-%d")
|
||||||
time_str = end_time.strftime("%H:%M:%S")
|
time_str = end_time.strftime("%H:%M:%S")
|
||||||
self.logger.info(f" {line_code}: {date_str} {time_str}")
|
self.logger.info(f" {line_code}: {date_str} {time_str}")
|
||||||
else:
|
else:
|
||||||
self.logger.info(" 全局字典为空")
|
self.logger.info(" 全局字典为空")
|
||||||
self.logger.info(f"总计: {len(global_variable.LINE_TIME_MAPPING_DICT)} 条记录\n")
|
self.logger.info(f"总计: {len(global_variable.get_line_time_mapping_dict())} 条记录\n")
|
||||||
def clear_line_time_mapping(self):
|
def clear_line_time_mapping(self):
|
||||||
"""
|
"""
|
||||||
清空全局字典
|
清空全局字典
|
||||||
"""
|
"""
|
||||||
global_variable.LINE_TIME_MAPPING_DICT.clear()
|
global_variable.get_line_time_mapping_dict().clear()
|
||||||
self.logger.info("已清空全局字典")
|
self.logger.info("已清空全局字典")
|
||||||
|
|
||||||
def set_device_time(self, device_id, time_str=None, date_str=None, disable_auto_sync=True):
|
def set_device_time(self, device_id, time_str=None, date_str=None, disable_auto_sync=True):
|
||||||
@@ -1121,10 +1121,10 @@ class ScreenshotPage:
|
|||||||
# 确保device_id正确设置,使用全局变量作为备用
|
# 确保device_id正确设置,使用全局变量作为备用
|
||||||
if not hasattr(self, 'device_id') or not self.device_id:
|
if not hasattr(self, 'device_id') or not self.device_id:
|
||||||
# 优先使用传入的device_id,其次使用全局变量
|
# 优先使用传入的device_id,其次使用全局变量
|
||||||
self.device_id = device_id if device_id else global_variable.GLOBAL_DEVICE_ID
|
self.device_id = device_id if device_id else global_variable.get_device_id()
|
||||||
|
|
||||||
# 使用self.device_id,确保有默认值
|
# 使用self.device_id,确保有默认值
|
||||||
actual_device_id = self.device_id if self.device_id else global_variable.GLOBAL_DEVICE_ID
|
actual_device_id = self.device_id if self.device_id else global_variable.get_device_id()
|
||||||
|
|
||||||
if not check_session_valid(self.driver, actual_device_id):
|
if not check_session_valid(self.driver, actual_device_id):
|
||||||
self.logger.warning(f"设备 {actual_device_id} 会话无效,尝试重新连接驱动...")
|
self.logger.warning(f"设备 {actual_device_id} 会话无效,尝试重新连接驱动...")
|
||||||
@@ -1269,9 +1269,9 @@ class ScreenshotPage:
|
|||||||
|
|
||||||
def add_breakpoint_to_upload_list(self, breakpoint_name, line_num):
|
def add_breakpoint_to_upload_list(self, breakpoint_name, line_num):
|
||||||
"""添加平差完成的断点到上传列表和字典"""
|
"""添加平差完成的断点到上传列表和字典"""
|
||||||
if breakpoint_name and breakpoint_name not in global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST:
|
if breakpoint_name and breakpoint_name not in global_variable.get_upload_breakpoint_list():
|
||||||
global_variable.GLOBAL_UPLOAD_BREAKPOINT_LIST.append(breakpoint_name)
|
global_variable.get_upload_breakpoint_list().append(breakpoint_name)
|
||||||
global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT[breakpoint_name] = {
|
global_variable.get_upload_breakpoint_dict()[breakpoint_name] = {
|
||||||
'breakpoint_name': breakpoint_name,
|
'breakpoint_name': breakpoint_name,
|
||||||
'line_num': line_num
|
'line_num': line_num
|
||||||
}
|
}
|
||||||
@@ -1342,19 +1342,18 @@ class ScreenshotPage:
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
# 检查GLOBAL_UPLOAD_BREAKPOINT_DICT是否为空,如果为空则初始化一些测试数据
|
# 检查GLOBAL_UPLOAD_BREAKPOINT_DICT是否为空,如果为空则初始化一些测试数据
|
||||||
if not global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT:
|
if not global_variable.get_upload_breakpoint_dict():
|
||||||
self.logger.warning("global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT为空,正在初始化测试数据")
|
self.logger.warning("global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT为空,正在初始化测试数据")
|
||||||
global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT = {'CDWZQ-2标-龙骨湾右线大桥-0-7号墩-平原': 'L156372', 'CDWZQ-2标-蓝家湾特大 桥-31-31-平原': 'L159206'}
|
global_variable.set_upload_breakpoint_dict({'CDWZQ-2标-龙骨湾右线大桥-0-7号墩-平原': 'L156372', 'CDWZQ-2标-蓝家湾特大 桥-31-31-平原': 'L159206'})
|
||||||
|
|
||||||
# 创建断点列表的副本,用于重试时重新处理
|
breakpoint_names = list(global_variable.get_upload_breakpoint_dict().keys())
|
||||||
breakpoint_names = list(global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT.keys())
|
|
||||||
processed_breakpoints = []
|
processed_breakpoints = []
|
||||||
|
|
||||||
# 开始循环处理断点
|
# 开始循环处理断点
|
||||||
for breakpoint_name in breakpoint_names:
|
for breakpoint_name in breakpoint_names:
|
||||||
if breakpoint_name in processed_breakpoints:
|
if breakpoint_name in processed_breakpoints:
|
||||||
continue
|
continue
|
||||||
line_code = global_variable.GLOBAL_UPLOAD_BREAKPOINT_DICT[breakpoint_name]
|
line_code = global_variable.get_upload_breakpoint_dict()[breakpoint_name]
|
||||||
|
|
||||||
self.logger.info(f"开始处理要平差的断点 {breakpoint_name}")
|
self.logger.info(f"开始处理要平差的断点 {breakpoint_name}")
|
||||||
|
|
||||||
@@ -1409,17 +1408,19 @@ class ScreenshotPage:
|
|||||||
self.logger.error(f"设备 {device_id} 处理返回按钮确认失败")
|
self.logger.error(f"设备 {device_id} 处理返回按钮确认失败")
|
||||||
continue
|
continue
|
||||||
|
|
||||||
# 成功处理完一个断点,添加到已处理列表
|
# # 成功处理完一个断点,添加到已处理列表
|
||||||
processed_breakpoints.append(breakpoint_name)
|
# processed_breakpoints.append(breakpoint_name)
|
||||||
self.logger.info(f"成功处理断点: {breakpoint_name}")
|
# self.logger.info(f"成功处理断点: {breakpoint_name}")
|
||||||
|
|
||||||
# 检查是否所有断点都处理完成
|
# # 检查是否所有断点都处理完成
|
||||||
if len(processed_breakpoints) == len(breakpoint_names):
|
# if len(processed_breakpoints) == len(breakpoint_names):
|
||||||
self.logger.info(f"设备 {device_id} 平差页面操作执行完成")
|
# self.logger.info(f"设备 {device_id} 平差页面操作执行完成")
|
||||||
return True
|
# return True
|
||||||
else:
|
# else:
|
||||||
self.logger.warning(f"设备 {device_id} 部分断点处理失败,已成功处理 {len(processed_breakpoints)}/{len(breakpoint_names)} 个断点")
|
# self.logger.warning(f"设备 {device_id} 部分断点处理失败,已成功处理 {len(processed_breakpoints)}/{len(breakpoint_names)} 个断点")
|
||||||
return True
|
# return True
|
||||||
|
self.logger.warning(f"设备 {device_id} 平差线路到上传页面流程执行完成")
|
||||||
|
return True
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
retry_count += 1
|
retry_count += 1
|
||||||
|
|||||||
@@ -671,7 +671,7 @@ class UploadConfigPage:
|
|||||||
logging.warning(f"跳过无效数据: 姓名='{name}', 身份证='{id_card}'")
|
logging.warning(f"跳过无效数据: 姓名='{name}', 身份证='{id_card}'")
|
||||||
|
|
||||||
# 将字典保存到全局变量
|
# 将字典保存到全局变量
|
||||||
global_variable.GLOBAL_NAME_TO_ID_MAP = name_id_map
|
global_variable.set_name_to_id_map(name_id_map)
|
||||||
|
|
||||||
logging.info(f"成功加载用户数据,共 {len(df)} 条记录,{len(name_id_map)} 个有效姓名-身份证映射")
|
logging.info(f"成功加载用户数据,共 {len(df)} 条记录,{len(name_id_map)} 个有效姓名-身份证映射")
|
||||||
|
|
||||||
@@ -711,7 +711,7 @@ class UploadConfigPage:
|
|||||||
logging.info(f"使用第一个数据员: {sjname}")
|
logging.info(f"使用第一个数据员: {sjname}")
|
||||||
|
|
||||||
# 获取身份证号码
|
# 获取身份证号码
|
||||||
id_card = global_variable.GLOBAL_NAME_TO_ID_MAP.get(sjname)
|
id_card = global_variable.get_name_to_id_map().get(sjname)
|
||||||
logging.info(f"id_card: {id_card}")
|
logging.info(f"id_card: {id_card}")
|
||||||
if not id_card:
|
if not id_card:
|
||||||
logging.error(f"未找到数据员 {sjname} 对应的身份证号")
|
logging.error(f"未找到数据员 {sjname} 对应的身份证号")
|
||||||
@@ -1498,6 +1498,7 @@ class UploadConfigPage:
|
|||||||
if not self.is_on_upload_config_page():
|
if not self.is_on_upload_config_page():
|
||||||
self.logger.info("不在上传配置页面,尝试导航...")
|
self.logger.info("不在上传配置页面,尝试导航...")
|
||||||
return False
|
return False
|
||||||
|
print(f"进入: {breakpoint_name}的上传配置页面")
|
||||||
|
|
||||||
# 检查是否有变化量属性
|
# 检查是否有变化量属性
|
||||||
if not self.check_change_amount_on_page():
|
if not self.check_change_amount_on_page():
|
||||||
@@ -1759,7 +1760,7 @@ class UploadConfigPage:
|
|||||||
else:
|
else:
|
||||||
self.logger.info("页面中包含变化量属性,继续执行后续操作")
|
self.logger.info("页面中包含变化量属性,继续执行后续操作")
|
||||||
|
|
||||||
user_id = global_variable.GLOBAL_USERNAME
|
user_id = global_variable.get_username()
|
||||||
if user_id is None:
|
if user_id is None:
|
||||||
self.logger.error("获取用户ID失败")
|
self.logger.error("获取用户ID失败")
|
||||||
return False
|
return False
|
||||||
@@ -1844,13 +1845,14 @@ class UploadConfigPage:
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
# # 表达填写完成,点击"保存上传"并处理弹窗
|
# 表达填写完成,点击"保存上传"并处理弹窗
|
||||||
# if not self.click_save_upload_and_handle_dialogs():
|
if not self.click_save_upload_and_handle_dialogs():
|
||||||
# self.logger.error("点击保存上传并处理弹窗失败")
|
self.logger.error("点击保存上传并处理弹窗失败")
|
||||||
# return False
|
return False
|
||||||
|
|
||||||
# 暂不上传,使用返回按钮替代。
|
# # 暂不上传,使用返回按钮替代。
|
||||||
self.driver.back()
|
# self.driver.back()
|
||||||
|
|
||||||
|
|
||||||
# 等待上传,查看loading弹窗。没有就下一个
|
# 等待上传,查看loading弹窗。没有就下一个
|
||||||
if not self.wait_for_upload_completion():
|
if not self.wait_for_upload_completion():
|
||||||
@@ -1859,7 +1861,7 @@ class UploadConfigPage:
|
|||||||
|
|
||||||
self.logger.info("上传配置页面操作执行完成")
|
self.logger.info("上传配置页面操作执行完成")
|
||||||
# 把上传成功的断点写入全局变量GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST
|
# 把上传成功的断点写入全局变量GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST
|
||||||
global_variable.GLOBAL_UPLOAD_SUCCESS_BREAKPOINT_LIST.append(breakpoint_name)
|
global_variable.get_upload_success_breakpoint_list().append(breakpoint_name)
|
||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|||||||
196
permissions.py
196
permissions.py
@@ -2,6 +2,7 @@
|
|||||||
import subprocess
|
import subprocess
|
||||||
import logging
|
import logging
|
||||||
import time
|
import time
|
||||||
|
import os
|
||||||
|
|
||||||
# 配置日志
|
# 配置日志
|
||||||
logging.basicConfig(level=logging.INFO, format="%(asctime)s - %(levelname)s: %(message)s")
|
logging.basicConfig(level=logging.INFO, format="%(asctime)s - %(levelname)s: %(message)s")
|
||||||
@@ -83,90 +84,143 @@ def grant_single_permission(device_id: str, package: str, permission: str) -> bo
|
|||||||
logging.error(f"设备 {device_id}:处理 {package} 时发生未知错误:{str(e)}")
|
logging.error(f"设备 {device_id}:处理 {package} 时发生未知错误:{str(e)}")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
# def grant_appium_permissions(device_id: str, require_all: bool = False) -> bool:
|
||||||
|
# """
|
||||||
|
# 为 Appium UiAutomator2 服务授予权限
|
||||||
|
# :param device_id: 设备 ID
|
||||||
|
# :param require_all: 是否要求所有权限都成功授予
|
||||||
|
# :return: 权限授予是否成功(根据require_all参数判断)
|
||||||
|
# """
|
||||||
|
# # 首先检查设备连接
|
||||||
|
# if not check_device_connection(device_id):
|
||||||
|
# return False
|
||||||
|
|
||||||
|
# packages_to_grant = [
|
||||||
|
# "io.appium.settings",
|
||||||
|
# "io.appium.uiautomator2.server",
|
||||||
|
# "io.appium.uiautomator2.server.test"
|
||||||
|
# ]
|
||||||
|
|
||||||
|
# # 权限列表(按优先级排序)
|
||||||
|
# permissions_to_grant = [
|
||||||
|
# "android.permission.SET_ANIMATION_SCALE",
|
||||||
|
# "android.permission.CHANGE_CONFIGURATION",
|
||||||
|
# "android.permission.WRITE_SETTINGS",
|
||||||
|
# "android.permission.DISABLE_KEYGUARD",
|
||||||
|
# ]
|
||||||
|
|
||||||
|
# success_count = 0
|
||||||
|
# total_attempted = 0
|
||||||
|
# package_results = {}
|
||||||
|
|
||||||
|
# # 检查并授予权限
|
||||||
|
# for package in packages_to_grant:
|
||||||
|
# package_results[package] = {"installed": False, "permissions": {}}
|
||||||
|
|
||||||
|
# if not is_package_installed(device_id, package):
|
||||||
|
# logging.warning(f"设备 {device_id}:包 {package} 未安装,跳过权限授予")
|
||||||
|
# package_results[package]["installed"] = False
|
||||||
|
# continue
|
||||||
|
|
||||||
|
# package_results[package]["installed"] = True
|
||||||
|
# package_success = 0
|
||||||
|
# package_attempted = 0
|
||||||
|
|
||||||
|
# for permission in permissions_to_grant:
|
||||||
|
# total_attempted += 1
|
||||||
|
# package_attempted += 1
|
||||||
|
|
||||||
|
# result = grant_single_permission(device_id, package, permission)
|
||||||
|
# package_results[package]["permissions"][permission] = result
|
||||||
|
|
||||||
|
# if result:
|
||||||
|
# success_count += 1
|
||||||
|
# package_success += 1
|
||||||
|
|
||||||
|
# # 记录每个包的授权结果
|
||||||
|
# logging.info(f"设备 {device_id}:包 {package} 权限授予结果: {package_success}/{package_attempted}")
|
||||||
|
|
||||||
|
# # 统计和报告
|
||||||
|
# logging.info(f"设备 {device_id}:权限授予完成")
|
||||||
|
# logging.info(f"总计: 尝试 {total_attempted} 次,成功 {success_count} 次")
|
||||||
|
|
||||||
|
# # 检查每个包的关键权限
|
||||||
|
# critical_permission = "android.permission.WRITE_SECURE_SETTINGS"
|
||||||
|
# critical_failures = []
|
||||||
|
|
||||||
|
# for package, info in package_results.items():
|
||||||
|
# if info["installed"] and critical_permission in info["permissions"]:
|
||||||
|
# if not info["permissions"][critical_permission]:
|
||||||
|
# critical_failures.append(package)
|
||||||
|
|
||||||
|
# if critical_failures:
|
||||||
|
# logging.warning(f"设备 {device_id}:以下包的关键权限 {critical_permission} 授予失败: {', '.join(critical_failures)}")
|
||||||
|
# logging.warning("这可能会影响某些自动化功能,但基础测试通常不受影响")
|
||||||
|
|
||||||
|
# # 根据require_all参数返回结果
|
||||||
|
# if require_all:
|
||||||
|
# # 要求所有权限都成功
|
||||||
|
# if critical_failures:
|
||||||
|
# logging.error("关键权限授予失败,无法继续(require_all=True)")
|
||||||
|
# return False
|
||||||
|
# return success_count == total_attempted
|
||||||
|
# else:
|
||||||
|
# # 不要求所有权限,只要设备连接正常就返回True
|
||||||
|
# logging.info(f"设备 {device_id}:权限授予过程完成,建议重启设备或Appium服务使更改生效")
|
||||||
|
# return True
|
||||||
|
|
||||||
def grant_appium_permissions(device_id: str, require_all: bool = False) -> bool:
|
def grant_appium_permissions(device_id: str, require_all: bool = False) -> bool:
|
||||||
"""
|
"""
|
||||||
为 Appium UiAutomator2 服务授予权限
|
修复版:为 Appium 授予权限(使用正确的方法)
|
||||||
:param device_id: 设备 ID
|
|
||||||
:param require_all: 是否要求所有权限都成功授予
|
|
||||||
:return: 权限授予是否成功(根据require_all参数判断)
|
|
||||||
"""
|
"""
|
||||||
# 首先检查设备连接
|
logging.info(f"设备 {device_id}:开始设置Appium权限")
|
||||||
if not check_device_connection(device_id):
|
|
||||||
return False
|
|
||||||
|
|
||||||
packages_to_grant = [
|
# 1. 使用系统设置命令(替代原来的pm grant尝试)
|
||||||
"io.appium.settings",
|
logging.info("使用系统设置命令...")
|
||||||
"io.appium.uiautomator2.server",
|
system_commands = [
|
||||||
"io.appium.uiautomator2.server.test"
|
["adb", "-s", device_id, "shell", "settings", "put", "global", "window_animation_scale", "0"],
|
||||||
]
|
["adb", "-s", device_id, "shell", "settings", "put", "global", "transition_animation_scale", "0"],
|
||||||
|
["adb", "-s", device_id, "shell", "settings", "put", "global", "animator_duration_scale", "0"],
|
||||||
# 权限列表(按优先级排序)
|
["adb", "-s", device_id, "shell", "settings", "put", "system", "screen_off_timeout", "86400000"],
|
||||||
permissions_to_grant = [
|
|
||||||
"android.permission.WRITE_SECURE_SETTINGS",
|
|
||||||
"android.permission.CHANGE_CONFIGURATION",
|
|
||||||
"android.permission.DUMP",
|
|
||||||
]
|
]
|
||||||
|
|
||||||
success_count = 0
|
success_count = 0
|
||||||
total_attempted = 0
|
for cmd in system_commands:
|
||||||
package_results = {}
|
try:
|
||||||
|
result = subprocess.run(cmd, capture_output=True, text=True, timeout=10)
|
||||||
# 检查并授予权限
|
if result.returncode == 0:
|
||||||
for package in packages_to_grant:
|
|
||||||
package_results[package] = {"installed": False, "permissions": {}}
|
|
||||||
|
|
||||||
if not is_package_installed(device_id, package):
|
|
||||||
logging.warning(f"设备 {device_id}:包 {package} 未安装,跳过权限授予")
|
|
||||||
package_results[package]["installed"] = False
|
|
||||||
continue
|
|
||||||
|
|
||||||
package_results[package]["installed"] = True
|
|
||||||
package_success = 0
|
|
||||||
package_attempted = 0
|
|
||||||
|
|
||||||
for permission in permissions_to_grant:
|
|
||||||
total_attempted += 1
|
|
||||||
package_attempted += 1
|
|
||||||
|
|
||||||
result = grant_single_permission(device_id, package, permission)
|
|
||||||
package_results[package]["permissions"][permission] = result
|
|
||||||
|
|
||||||
if result:
|
|
||||||
success_count += 1
|
success_count += 1
|
||||||
package_success += 1
|
logging.info(f" 成功: {' '.join(cmd[3:])}")
|
||||||
|
else:
|
||||||
# 记录每个包的授权结果
|
logging.warning(f" 失败: {' '.join(cmd[3:])}")
|
||||||
logging.info(f"设备 {device_id}:包 {package} 权限授予结果: {package_success}/{package_attempted}")
|
except:
|
||||||
|
logging.warning(f" 异常: {' '.join(cmd[3:])}")
|
||||||
|
|
||||||
# 统计和报告
|
# 2. 授予可自动授予的权限
|
||||||
logging.info(f"设备 {device_id}:权限授予完成")
|
logging.info("授予基础权限...")
|
||||||
logging.info(f"总计: 尝试 {total_attempted} 次,成功 {success_count} 次")
|
grantable = [
|
||||||
|
"android.permission.INTERNET",
|
||||||
|
"android.permission.ACCESS_NETWORK_STATE",
|
||||||
|
"android.permission.ACCESS_WIFI_STATE",
|
||||||
|
]
|
||||||
|
|
||||||
# 检查每个包的关键权限
|
for perm in grantable:
|
||||||
critical_permission = "android.permission.WRITE_SECURE_SETTINGS"
|
cmd = ["adb", "-s", device_id, "shell", "pm", "grant", "io.appium.settings", perm]
|
||||||
critical_failures = []
|
result = subprocess.run(cmd, capture_output=True, text=True, timeout=10)
|
||||||
|
if result.returncode == 0:
|
||||||
|
success_count += 1
|
||||||
|
logging.info(f" 成功授予: {perm.split('.')[-1]}")
|
||||||
|
else:
|
||||||
|
logging.debug(f" 跳过: {perm.split('.')[-1]}")
|
||||||
|
|
||||||
for package, info in package_results.items():
|
# 3. 返回结果
|
||||||
if info["installed"] and critical_permission in info["permissions"]:
|
logging.info(f"设置完成,成功项数: {success_count}")
|
||||||
if not info["permissions"][critical_permission]:
|
|
||||||
critical_failures.append(package)
|
|
||||||
|
|
||||||
if critical_failures:
|
|
||||||
logging.warning(f"设备 {device_id}:以下包的关键权限 {critical_permission} 授予失败: {', '.join(critical_failures)}")
|
|
||||||
logging.warning("这可能会影响某些自动化功能,但基础测试通常不受影响")
|
|
||||||
|
|
||||||
# 根据require_all参数返回结果
|
|
||||||
if require_all:
|
if require_all:
|
||||||
# 要求所有权限都成功
|
return success_count == (len(system_commands) + len(grantable))
|
||||||
if critical_failures:
|
|
||||||
logging.error("关键权限授予失败,无法继续(require_all=True)")
|
|
||||||
return False
|
|
||||||
return success_count == total_attempted
|
|
||||||
else:
|
else:
|
||||||
# 不要求所有权限,只要设备连接正常就返回True
|
return success_count > 0 # 只要有成功项就返回True
|
||||||
logging.info(f"设备 {device_id}:权限授予过程完成,建议重启设备或Appium服务使更改生效")
|
|
||||||
return True
|
|
||||||
|
|
||||||
def check_appium_compatibility(device_id: str) -> dict:
|
def check_appium_compatibility(device_id: str) -> dict:
|
||||||
"""
|
"""
|
||||||
检查Appium兼容性
|
检查Appium兼容性
|
||||||
@@ -217,6 +271,8 @@ def check_appium_compatibility(device_id: str) -> dict:
|
|||||||
logging.error(f"检查兼容性时出错: {str(e)}")
|
logging.error(f"检查兼容性时出错: {str(e)}")
|
||||||
return {"device_id": device_id, "error": str(e)}
|
return {"device_id": device_id, "error": str(e)}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# 使用示例
|
# 使用示例
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
# 获取设备ID(示例)
|
# 获取设备ID(示例)
|
||||||
|
|||||||
2
requirements.txt
Normal file
2
requirements.txt
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
appium-python-client
|
||||||
|
selenium
|
||||||
185
scheduler.py
Normal file
185
scheduler.py
Normal file
@@ -0,0 +1,185 @@
|
|||||||
|
import requests
|
||||||
|
import time
|
||||||
|
from concurrent.futures import ThreadPoolExecutor, as_completed
|
||||||
|
|
||||||
|
from main import DeviceAutomation
|
||||||
|
import globals.apis as apis
|
||||||
|
import random
|
||||||
|
from datetime import datetime, timedelta
|
||||||
|
|
||||||
|
## --- 配置区 ---
|
||||||
|
API_URL = "http://your-api-server.com/api/tasks" # 替换为真实的接口地址
|
||||||
|
MAX_WORKERS = 3 # 最大并发线程数,建议根据网络带宽调整
|
||||||
|
TIME_FILE_PATH = r"D:\time.txt" # 电脑D盘下的路径
|
||||||
|
|
||||||
|
def parse_time_config():
|
||||||
|
"""
|
||||||
|
解析 D:\time.txt 文件
|
||||||
|
返回格式: { "用户名": "16:40:20", ... }
|
||||||
|
"""
|
||||||
|
time_map = {}
|
||||||
|
if not os.path.exists(TIME_FILE_PATH):
|
||||||
|
print(f"⚠️ 未找到配置文件: {TIME_FILE_PATH}")
|
||||||
|
return time_map
|
||||||
|
|
||||||
|
try:
|
||||||
|
with open(TIME_FILE_PATH, 'r', encoding='utf-8') as f:
|
||||||
|
for line in f:
|
||||||
|
line = line.strip()
|
||||||
|
if not line:
|
||||||
|
continue
|
||||||
|
|
||||||
|
# 使用正则匹配用户名、时间、状态
|
||||||
|
# 兼容 wangshun 16:40:20 true 和 cdwzq3liangchaoyong 15:06:35 true
|
||||||
|
match = re.search(r'(\w+)\s+(\d{2}:\d{2}:\d{2})\s+true', line)
|
||||||
|
if match:
|
||||||
|
username = match.group(1)
|
||||||
|
scheduled_time = match.group(2)
|
||||||
|
time_map[username] = scheduled_time
|
||||||
|
except Exception as e:
|
||||||
|
print(f"❌ 解析 time.txt 失败: {e}")
|
||||||
|
|
||||||
|
return time_map
|
||||||
|
|
||||||
|
def get_remote_tasks():
|
||||||
|
"""
|
||||||
|
从接口获取数据
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
# 1. 先获取本地文件中的配置
|
||||||
|
local_times = parse_time_config()
|
||||||
|
if not local_times:
|
||||||
|
print("❌ time.txt 中没有有效的 true 任务或文件为空")
|
||||||
|
return {}
|
||||||
|
|
||||||
|
# 2. 从服务器获取账户
|
||||||
|
accounts = apis.get_accounts_from_server("68c0dbfdb7cbcd616e7c5ab5")
|
||||||
|
if not accounts:
|
||||||
|
print("❌ 未从服务器获取到账户信息,终止流程")
|
||||||
|
return {}
|
||||||
|
|
||||||
|
# filtered_accounts = [account for account in accounts if account.get('is_ok') == 1]
|
||||||
|
# # print("✅ 获取账户信息成功", filtered_accounts)
|
||||||
|
# # current_time = datetime.now().strftime("%Y-%m-%d %H:%M:%S")
|
||||||
|
|
||||||
|
# # 从原始 accounts 数据中筛选有 device_ip 和 device_port 且不为 None 的账户
|
||||||
|
# device_dict = {}
|
||||||
|
|
||||||
|
# for account in filtered_accounts:
|
||||||
|
# device_ip = account.get('device_ip')
|
||||||
|
# device_port = account.get('device_port')
|
||||||
|
|
||||||
|
# # 检查 device_ip 和 device_port 是否都存在且不为 None
|
||||||
|
# if device_ip and device_port:
|
||||||
|
# # 拼接为 ip:port 格式
|
||||||
|
# key = f"{device_ip}:{device_port}"
|
||||||
|
# # 添加当前时间
|
||||||
|
# # device_dict[key] = current_time
|
||||||
|
# # 获取当前时间的 datetime 对象
|
||||||
|
# current_datetime = datetime.now()
|
||||||
|
|
||||||
|
# # 生成1-2小时的随机时间
|
||||||
|
# random_hours = random.uniform(1, 2)
|
||||||
|
|
||||||
|
# # 计算未来时间(datetime 对象可以加 timedelta)
|
||||||
|
# future_datetime = current_datetime + timedelta(hours=random_hours)
|
||||||
|
|
||||||
|
# # 将 datetime 对象格式化为字符串
|
||||||
|
# time_str = future_datetime.strftime("%Y-%m-%d %H:%M:%S")
|
||||||
|
|
||||||
|
# # 添加时间字符串到字典
|
||||||
|
# device_dict[key] = time_str
|
||||||
|
|
||||||
|
|
||||||
|
# # 结果
|
||||||
|
# print(device_dict)
|
||||||
|
# return device_dict
|
||||||
|
device_dict = {}
|
||||||
|
today_str = datetime.now().strftime("%Y-%m-%d")
|
||||||
|
|
||||||
|
for account in accounts:
|
||||||
|
# 条件1: 接口返回 is_ok == 1
|
||||||
|
if account.get('is_ok') != 1:
|
||||||
|
continue
|
||||||
|
|
||||||
|
username = account.get('username') # 假设接口中用户名的 key 是 username
|
||||||
|
device_ip = account.get('device_ip')
|
||||||
|
device_port = account.get('device_port')
|
||||||
|
|
||||||
|
# 条件2: 用户名在 time.txt 中且状态为 true (已经在 local_times 过滤过)
|
||||||
|
if username in local_times and device_ip and device_port:
|
||||||
|
key = f"{device_ip}:{device_port}"
|
||||||
|
# 拼接成完整的 YYYY-MM-DD HH:MM:SS
|
||||||
|
full_time_str = f"{today_str} {local_times[username]}"
|
||||||
|
device_dict[key] = full_time_str
|
||||||
|
|
||||||
|
print(f"✅ 匹配成功,待执行任务详情: {device_dict}")
|
||||||
|
return device_dict
|
||||||
|
# # 模拟数据供测试
|
||||||
|
# return {
|
||||||
|
# "192.168.1.45:5555": "2026-02-03 10:20:00",
|
||||||
|
# "192.168.1.100:5556": "2026-02-03 10:20:20",
|
||||||
|
# "192.168.31.12:5557": "2026-02-03 10:21:00"
|
||||||
|
# }
|
||||||
|
except Exception as e:
|
||||||
|
print(f"❌ 获取接口任务失败: {e}")
|
||||||
|
return {}
|
||||||
|
|
||||||
|
def run_task(address, target_time):
|
||||||
|
"""
|
||||||
|
单个线程的任务包装器
|
||||||
|
"""
|
||||||
|
# 格式化账号/设备名:确保带上端口号
|
||||||
|
device_address = address
|
||||||
|
|
||||||
|
print(f"🕒 [等待/启动] 设备: {device_address} | 预定时间: {target_time}")
|
||||||
|
# 解析目标时间
|
||||||
|
target_datetime = datetime.strptime(target_time, "%Y-%m-%d %H:%M:%S")
|
||||||
|
current_datetime = datetime.now()
|
||||||
|
|
||||||
|
# 计算等待时间
|
||||||
|
wait_seconds = (target_datetime - current_datetime).total_seconds()
|
||||||
|
|
||||||
|
# 如果需要等待
|
||||||
|
if wait_seconds > 0:
|
||||||
|
print(f"⏳ [等待中] 设备: {device_address} | 等待时间: {wait_seconds:.2f}秒")
|
||||||
|
time.sleep(wait_seconds)
|
||||||
|
print(f"▶️ [开始执行] 设备: {device_address} | 当前时间: {datetime.now().strftime('%Y-%m-%d %H:%M:%S')}")
|
||||||
|
else:
|
||||||
|
print(f"▶️ [开始执行] 设备: {device_address} | 当前时间: {datetime.now().strftime('%Y-%m-%d %H:%M:%S')}")
|
||||||
|
|
||||||
|
|
||||||
|
try:
|
||||||
|
# 创建DeviceAutomation实例并执行上传逻辑
|
||||||
|
automation = DeviceAutomation(device_address)
|
||||||
|
result = automation.handle_app_state()
|
||||||
|
return f"✅ {device_address} 完成: {result}"
|
||||||
|
except Exception as e:
|
||||||
|
return f"❌ {device_address} 报错: {str(e)}"
|
||||||
|
|
||||||
|
def monitor_center():
|
||||||
|
"""调度中心"""
|
||||||
|
tasks_data = get_remote_tasks()
|
||||||
|
|
||||||
|
if not tasks_data:
|
||||||
|
print("📭 接口未返回任何任务,程序退出。")
|
||||||
|
return
|
||||||
|
|
||||||
|
print(f"🗂️ 发现 {len(tasks_data)} 个待处理账号,开始建立线程池...")
|
||||||
|
|
||||||
|
# 使用线程池并发执行
|
||||||
|
with ThreadPoolExecutor(max_workers=MAX_WORKERS) as executor:
|
||||||
|
# 提交所有任务
|
||||||
|
future_to_device = {
|
||||||
|
executor.submit(run_task, acc, t): acc
|
||||||
|
for acc, t in tasks_data.items()
|
||||||
|
}
|
||||||
|
|
||||||
|
# 实时打印完成情况
|
||||||
|
for future in as_completed(future_to_device):
|
||||||
|
print(future.result())
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
print("🚀 自动化调度程序启动...")
|
||||||
|
monitor_center()
|
||||||
|
print("🏁 所有并发任务处理序列结束。")
|
||||||
193
scheduler_two.py
Normal file
193
scheduler_two.py
Normal file
@@ -0,0 +1,193 @@
|
|||||||
|
import time
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
import threading
|
||||||
|
from concurrent.futures import ThreadPoolExecutor, as_completed
|
||||||
|
from datetime import datetime
|
||||||
|
from main import DeviceAutomation
|
||||||
|
import globals.apis as apis
|
||||||
|
|
||||||
|
## --- 配置区 ---
|
||||||
|
API_URL = "http://your-api-server.com/api/tasks"
|
||||||
|
MAX_WORKERS = 3
|
||||||
|
TIME_FILE_PATH = r"D:\time.txt"
|
||||||
|
POLLING_INTERVAL = 300 # 30分钟轮询一次
|
||||||
|
|
||||||
|
# 文件操作锁,防止多线程同时写入文件造成冲突
|
||||||
|
file_lock = threading.Lock()
|
||||||
|
|
||||||
|
def update_file_status(username, from_status, to_status):
|
||||||
|
"""
|
||||||
|
安全地更新 time.txt 中该用户的状态
|
||||||
|
例如: 将 'true' 改为 'running', 或将 'running' 改为 'done'
|
||||||
|
"""
|
||||||
|
if not os.path.exists(TIME_FILE_PATH):
|
||||||
|
return False
|
||||||
|
|
||||||
|
success = False
|
||||||
|
with file_lock:
|
||||||
|
try:
|
||||||
|
with open(TIME_FILE_PATH, 'r', encoding='utf-8') as f:
|
||||||
|
lines = f.readlines()
|
||||||
|
|
||||||
|
# new_lines = []
|
||||||
|
# for line in lines:
|
||||||
|
# clean_line = line.strip()
|
||||||
|
# # 匹配逻辑:包含用户名 且 以 from_status 结尾
|
||||||
|
# if f" {username} " in line and clean_line.endswith(from_status):
|
||||||
|
# line = line.replace(from_status, to_status)
|
||||||
|
# success = True
|
||||||
|
# new_lines.append(line)
|
||||||
|
new_lines = []
|
||||||
|
for line in lines:
|
||||||
|
# 使用正则确保精准匹配用户名和结尾状态
|
||||||
|
# 匹配规则:行内包含该用户名,且该行以 from_status 结尾
|
||||||
|
if re.search(rf'\b{username}\b', line) and line.strip().endswith(from_status):
|
||||||
|
# 只替换行尾的那个状态词
|
||||||
|
line = re.sub(rf'{from_status}$', to_status, line.rstrip()) + '\n'
|
||||||
|
success = True
|
||||||
|
new_lines.append(line)
|
||||||
|
|
||||||
|
with open(TIME_FILE_PATH, 'w', encoding='utf-8') as f:
|
||||||
|
f.writelines(new_lines)
|
||||||
|
|
||||||
|
if success:
|
||||||
|
print(f"📝 [文件更新] 用户 {username}: {from_status} -> {to_status}")
|
||||||
|
return success
|
||||||
|
except Exception as e:
|
||||||
|
print(f"❌ 更新文件状态失败 ({username}): {e}")
|
||||||
|
return False
|
||||||
|
|
||||||
|
def parse_time_config():
|
||||||
|
"""
|
||||||
|
解析 time.txt,只获取状态为 true 的任务
|
||||||
|
"""
|
||||||
|
time_map = {}
|
||||||
|
if not os.path.exists(TIME_FILE_PATH):
|
||||||
|
print(f"⚠️ 文件不存在: {TIME_FILE_PATH}")
|
||||||
|
return time_map
|
||||||
|
|
||||||
|
try:
|
||||||
|
with file_lock:
|
||||||
|
with open(TIME_FILE_PATH, 'r', encoding='utf-8') as f:
|
||||||
|
for line in f:
|
||||||
|
line = line.strip()
|
||||||
|
# 匹配:用户名 时间 true (仅获取待处理任务)
|
||||||
|
match = re.search(r'(\w+)\s+(\d{1,2}:\d{2}:\d{2})\s+true$', line)
|
||||||
|
if match:
|
||||||
|
username, scheduled_time = match.group(1), match.group(2)
|
||||||
|
time_map[username] = scheduled_time
|
||||||
|
except Exception as e:
|
||||||
|
print(f"❌ 解析 time.txt 失败: {e}")
|
||||||
|
return time_map
|
||||||
|
|
||||||
|
def get_combined_tasks():
|
||||||
|
"""
|
||||||
|
结合接口(is_ok==1)和本地文件(true)筛选任务
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
local_times = parse_time_config()
|
||||||
|
if not local_times:
|
||||||
|
return {}
|
||||||
|
|
||||||
|
# 调用你的 API 接口获取账号信息
|
||||||
|
accounts = apis.get_accounts_from_server("68c0dbfdb7cbcd616e7c5ab5")
|
||||||
|
if not accounts:
|
||||||
|
return {}
|
||||||
|
|
||||||
|
task_list = {}
|
||||||
|
today = datetime.now().strftime("%Y-%m-%d")
|
||||||
|
|
||||||
|
for account in accounts:
|
||||||
|
if account.get('is_ok') == 1:
|
||||||
|
user = account.get('username')
|
||||||
|
ip = account.get('device_ip')
|
||||||
|
port = account.get('device_port')
|
||||||
|
|
||||||
|
# 只有在 time.txt 中是 true 的账号才会被加入
|
||||||
|
if user in local_times and ip and port:
|
||||||
|
address = f"{ip}:{port}"
|
||||||
|
# full_time = f"{today} {local_times[user]}"
|
||||||
|
# 确保时间是两位数格式
|
||||||
|
raw_time = local_times[user]
|
||||||
|
# 将时间格式化为两位数:9:52:20 -> 09:52:20
|
||||||
|
if ':' in raw_time:
|
||||||
|
parts = raw_time.split(':')
|
||||||
|
if len(parts[0]) == 1:
|
||||||
|
raw_time = f"0{raw_time}" # 补齐前导零
|
||||||
|
|
||||||
|
full_time = f"{today} {raw_time}"
|
||||||
|
task_list[address] = {"time": full_time, "user": user}
|
||||||
|
|
||||||
|
return task_list
|
||||||
|
except Exception as e:
|
||||||
|
print(f"❌ 获取任务异常: {e}")
|
||||||
|
return {}
|
||||||
|
|
||||||
|
def run_task(address, target_time, username):
|
||||||
|
"""
|
||||||
|
单个执行线程:锁定状态 -> 等待 -> 执行 -> 完成
|
||||||
|
"""
|
||||||
|
# 1. 尝试将状态从 true 改为 running (锁定任务)
|
||||||
|
# 如果此时文件状态已被其他逻辑修改,则放弃执行,防止重复
|
||||||
|
if not update_file_status(username, "true", "running"):
|
||||||
|
return f"⏭️ {username} 状态已变更,跳过执行。"
|
||||||
|
|
||||||
|
print(f"🚀 [任务锁定] 设备: {address} | 用户: {username} | 计划时间: {target_time}")
|
||||||
|
|
||||||
|
try:
|
||||||
|
# 2. 计算并执行等待逻辑
|
||||||
|
target_dt = datetime.strptime(target_time, "%Y-%m-%d %H:%M:%S")
|
||||||
|
wait_secs = (target_dt - datetime.now()).total_seconds()
|
||||||
|
|
||||||
|
if wait_secs > 0:
|
||||||
|
print(f"⏳ {username} 距离执行还有 {int(wait_secs)} 秒...")
|
||||||
|
time.sleep(wait_secs)
|
||||||
|
|
||||||
|
# 3. 调用 main.py 中的自动化逻辑
|
||||||
|
print(f"▶️ [正在执行] {username} 开始自动化操作...")
|
||||||
|
automation = DeviceAutomation(address)
|
||||||
|
result = automation.handle_app_state()
|
||||||
|
|
||||||
|
# 4. 执行完成后,将状态从 running 改为 done
|
||||||
|
update_file_status(username, "running", "done")
|
||||||
|
return f"✅ {username} 执行成功: {result}"
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
# 如果中间报错,将状态改为 error 方便排查
|
||||||
|
update_file_status(username, "running", "error")
|
||||||
|
return f"❌ {username} 执行异常: {str(e)}"
|
||||||
|
|
||||||
|
def monitor_center():
|
||||||
|
"""调度中心:每半小时检查一次"""
|
||||||
|
while True:
|
||||||
|
now_str = datetime.now().strftime('%Y-%m-%d %H:%M:%S')
|
||||||
|
print(f"\n{'='*20} 周期性检查开始 ({now_str}) {'='*20}")
|
||||||
|
|
||||||
|
tasks = get_combined_tasks()
|
||||||
|
|
||||||
|
if tasks:
|
||||||
|
print(f"📡 发现 {len(tasks)} 个符合条件且未跑过的任务,准备启动线程池...")
|
||||||
|
with ThreadPoolExecutor(max_workers=MAX_WORKERS) as executor:
|
||||||
|
# 提交任务,将 address, time, username 传入
|
||||||
|
future_to_user = {
|
||||||
|
executor.submit(run_task, addr, info['time'], info['user']): info['user']
|
||||||
|
for addr, info in tasks.items()
|
||||||
|
}
|
||||||
|
|
||||||
|
for f in as_completed(future_to_user):
|
||||||
|
print(f.result())
|
||||||
|
else:
|
||||||
|
print("📭 暂无待处理任务 (所有账号已完成或 is_ok != 1)")
|
||||||
|
|
||||||
|
print(f"💤 本轮轮询结束,等待 {POLLING_INTERVAL//60} 分钟进行下次检查...")
|
||||||
|
time.sleep(POLLING_INTERVAL)
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
print("🌟 自动化调度服务已启动")
|
||||||
|
print(f"📂 配置文件: {TIME_FILE_PATH}")
|
||||||
|
print(f"⏱️ 轮询频率: {POLLING_INTERVAL}秒")
|
||||||
|
try:
|
||||||
|
monitor_center()
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
print("\n👋 用户手动停止程序。")
|
||||||
38
scheduler_two.spec
Normal file
38
scheduler_two.spec
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
# -*- mode: python ; coding: utf-8 -*-
|
||||||
|
|
||||||
|
|
||||||
|
a = Analysis(
|
||||||
|
['scheduler_two.py'],
|
||||||
|
pathex=[],
|
||||||
|
binaries=[],
|
||||||
|
datas=[],
|
||||||
|
hiddenimports=[],
|
||||||
|
hookspath=[],
|
||||||
|
hooksconfig={},
|
||||||
|
runtime_hooks=[],
|
||||||
|
excludes=[],
|
||||||
|
noarchive=False,
|
||||||
|
optimize=0,
|
||||||
|
)
|
||||||
|
pyz = PYZ(a.pure)
|
||||||
|
|
||||||
|
exe = EXE(
|
||||||
|
pyz,
|
||||||
|
a.scripts,
|
||||||
|
a.binaries,
|
||||||
|
a.datas,
|
||||||
|
[],
|
||||||
|
name='scheduler_two',
|
||||||
|
debug=False,
|
||||||
|
bootloader_ignore_signals=False,
|
||||||
|
strip=False,
|
||||||
|
upx=True,
|
||||||
|
upx_exclude=[],
|
||||||
|
runtime_tmpdir=None,
|
||||||
|
console=True,
|
||||||
|
disable_windowed_traceback=False,
|
||||||
|
argv_emulation=False,
|
||||||
|
target_arch=None,
|
||||||
|
codesign_identity=None,
|
||||||
|
entitlements_file=None,
|
||||||
|
)
|
||||||
@@ -1,5 +1,8 @@
|
|||||||
CDWZQ-3标-雷庙村大桥-8-号桥台-平原
|
CDWZQ-2标-蓝家湾特大桥-24-30-平原
|
||||||
CDWZQ-3标-雷庙村特大桥-5-6-号墩-平原
|
CDWZQ-2标-155号路基右线-461274-461570-155右-平原
|
||||||
CDWZQ-3标-金马村特大桥-14-号墩-平原
|
CDWZQ-2标-龙骨湾右线大桥-0-7号墩-平原
|
||||||
CDWZQ-3标-雷庙村特大桥-4#-7-14号墩-平原
|
CDWZQ-2标-区间路基135号-447663-447219-平原
|
||||||
CDWZQ-3标-老屋坡特大桥-14-21-平原
|
CDWZQ-2标-区间路基152号小-458515-459000-152-1-平原
|
||||||
|
CDWZQ-2标-区间路基137-449311-449487-137-平原
|
||||||
|
CDWZQ-2标-155号路基左线新-461121-461570-2-平原
|
||||||
|
CDWZQ-2标-蓝家湾特大桥-31-31-平原
|
||||||
|
|||||||
Reference in New Issue
Block a user