oscura 0.0.1__py3-none-any.whl → 0.1.1__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (465) hide show
  1. oscura/__init__.py +813 -8
  2. oscura/__main__.py +392 -0
  3. oscura/analyzers/__init__.py +37 -0
  4. oscura/analyzers/digital/__init__.py +177 -0
  5. oscura/analyzers/digital/bus.py +691 -0
  6. oscura/analyzers/digital/clock.py +805 -0
  7. oscura/analyzers/digital/correlation.py +720 -0
  8. oscura/analyzers/digital/edges.py +632 -0
  9. oscura/analyzers/digital/extraction.py +413 -0
  10. oscura/analyzers/digital/quality.py +878 -0
  11. oscura/analyzers/digital/signal_quality.py +877 -0
  12. oscura/analyzers/digital/thresholds.py +708 -0
  13. oscura/analyzers/digital/timing.py +1104 -0
  14. oscura/analyzers/eye/__init__.py +46 -0
  15. oscura/analyzers/eye/diagram.py +434 -0
  16. oscura/analyzers/eye/metrics.py +555 -0
  17. oscura/analyzers/jitter/__init__.py +83 -0
  18. oscura/analyzers/jitter/ber.py +333 -0
  19. oscura/analyzers/jitter/decomposition.py +759 -0
  20. oscura/analyzers/jitter/measurements.py +413 -0
  21. oscura/analyzers/jitter/spectrum.py +220 -0
  22. oscura/analyzers/measurements.py +40 -0
  23. oscura/analyzers/packet/__init__.py +171 -0
  24. oscura/analyzers/packet/daq.py +1077 -0
  25. oscura/analyzers/packet/metrics.py +437 -0
  26. oscura/analyzers/packet/parser.py +327 -0
  27. oscura/analyzers/packet/payload.py +2156 -0
  28. oscura/analyzers/packet/payload_analysis.py +1312 -0
  29. oscura/analyzers/packet/payload_extraction.py +236 -0
  30. oscura/analyzers/packet/payload_patterns.py +670 -0
  31. oscura/analyzers/packet/stream.py +359 -0
  32. oscura/analyzers/patterns/__init__.py +266 -0
  33. oscura/analyzers/patterns/clustering.py +1036 -0
  34. oscura/analyzers/patterns/discovery.py +539 -0
  35. oscura/analyzers/patterns/learning.py +797 -0
  36. oscura/analyzers/patterns/matching.py +1091 -0
  37. oscura/analyzers/patterns/periodic.py +650 -0
  38. oscura/analyzers/patterns/sequences.py +767 -0
  39. oscura/analyzers/power/__init__.py +116 -0
  40. oscura/analyzers/power/ac_power.py +391 -0
  41. oscura/analyzers/power/basic.py +383 -0
  42. oscura/analyzers/power/conduction.py +314 -0
  43. oscura/analyzers/power/efficiency.py +297 -0
  44. oscura/analyzers/power/ripple.py +356 -0
  45. oscura/analyzers/power/soa.py +372 -0
  46. oscura/analyzers/power/switching.py +479 -0
  47. oscura/analyzers/protocol/__init__.py +150 -0
  48. oscura/analyzers/protocols/__init__.py +150 -0
  49. oscura/analyzers/protocols/base.py +500 -0
  50. oscura/analyzers/protocols/can.py +620 -0
  51. oscura/analyzers/protocols/can_fd.py +448 -0
  52. oscura/analyzers/protocols/flexray.py +405 -0
  53. oscura/analyzers/protocols/hdlc.py +399 -0
  54. oscura/analyzers/protocols/i2c.py +368 -0
  55. oscura/analyzers/protocols/i2s.py +296 -0
  56. oscura/analyzers/protocols/jtag.py +393 -0
  57. oscura/analyzers/protocols/lin.py +445 -0
  58. oscura/analyzers/protocols/manchester.py +333 -0
  59. oscura/analyzers/protocols/onewire.py +501 -0
  60. oscura/analyzers/protocols/spi.py +334 -0
  61. oscura/analyzers/protocols/swd.py +325 -0
  62. oscura/analyzers/protocols/uart.py +393 -0
  63. oscura/analyzers/protocols/usb.py +495 -0
  64. oscura/analyzers/signal_integrity/__init__.py +63 -0
  65. oscura/analyzers/signal_integrity/embedding.py +294 -0
  66. oscura/analyzers/signal_integrity/equalization.py +370 -0
  67. oscura/analyzers/signal_integrity/sparams.py +484 -0
  68. oscura/analyzers/spectral/__init__.py +53 -0
  69. oscura/analyzers/spectral/chunked.py +273 -0
  70. oscura/analyzers/spectral/chunked_fft.py +571 -0
  71. oscura/analyzers/spectral/chunked_wavelet.py +391 -0
  72. oscura/analyzers/spectral/fft.py +92 -0
  73. oscura/analyzers/statistical/__init__.py +250 -0
  74. oscura/analyzers/statistical/checksum.py +923 -0
  75. oscura/analyzers/statistical/chunked_corr.py +228 -0
  76. oscura/analyzers/statistical/classification.py +778 -0
  77. oscura/analyzers/statistical/entropy.py +1113 -0
  78. oscura/analyzers/statistical/ngrams.py +614 -0
  79. oscura/analyzers/statistics/__init__.py +119 -0
  80. oscura/analyzers/statistics/advanced.py +885 -0
  81. oscura/analyzers/statistics/basic.py +263 -0
  82. oscura/analyzers/statistics/correlation.py +630 -0
  83. oscura/analyzers/statistics/distribution.py +298 -0
  84. oscura/analyzers/statistics/outliers.py +463 -0
  85. oscura/analyzers/statistics/streaming.py +93 -0
  86. oscura/analyzers/statistics/trend.py +520 -0
  87. oscura/analyzers/validation.py +598 -0
  88. oscura/analyzers/waveform/__init__.py +36 -0
  89. oscura/analyzers/waveform/measurements.py +943 -0
  90. oscura/analyzers/waveform/measurements_with_uncertainty.py +371 -0
  91. oscura/analyzers/waveform/spectral.py +1689 -0
  92. oscura/analyzers/waveform/wavelets.py +298 -0
  93. oscura/api/__init__.py +62 -0
  94. oscura/api/dsl.py +538 -0
  95. oscura/api/fluent.py +571 -0
  96. oscura/api/operators.py +498 -0
  97. oscura/api/optimization.py +392 -0
  98. oscura/api/profiling.py +396 -0
  99. oscura/automotive/__init__.py +73 -0
  100. oscura/automotive/can/__init__.py +52 -0
  101. oscura/automotive/can/analysis.py +356 -0
  102. oscura/automotive/can/checksum.py +250 -0
  103. oscura/automotive/can/correlation.py +212 -0
  104. oscura/automotive/can/discovery.py +355 -0
  105. oscura/automotive/can/message_wrapper.py +375 -0
  106. oscura/automotive/can/models.py +385 -0
  107. oscura/automotive/can/patterns.py +381 -0
  108. oscura/automotive/can/session.py +452 -0
  109. oscura/automotive/can/state_machine.py +300 -0
  110. oscura/automotive/can/stimulus_response.py +461 -0
  111. oscura/automotive/dbc/__init__.py +15 -0
  112. oscura/automotive/dbc/generator.py +156 -0
  113. oscura/automotive/dbc/parser.py +146 -0
  114. oscura/automotive/dtc/__init__.py +30 -0
  115. oscura/automotive/dtc/database.py +3036 -0
  116. oscura/automotive/j1939/__init__.py +14 -0
  117. oscura/automotive/j1939/decoder.py +745 -0
  118. oscura/automotive/loaders/__init__.py +35 -0
  119. oscura/automotive/loaders/asc.py +98 -0
  120. oscura/automotive/loaders/blf.py +77 -0
  121. oscura/automotive/loaders/csv_can.py +136 -0
  122. oscura/automotive/loaders/dispatcher.py +136 -0
  123. oscura/automotive/loaders/mdf.py +331 -0
  124. oscura/automotive/loaders/pcap.py +132 -0
  125. oscura/automotive/obd/__init__.py +14 -0
  126. oscura/automotive/obd/decoder.py +707 -0
  127. oscura/automotive/uds/__init__.py +48 -0
  128. oscura/automotive/uds/decoder.py +265 -0
  129. oscura/automotive/uds/models.py +64 -0
  130. oscura/automotive/visualization.py +369 -0
  131. oscura/batch/__init__.py +55 -0
  132. oscura/batch/advanced.py +627 -0
  133. oscura/batch/aggregate.py +300 -0
  134. oscura/batch/analyze.py +139 -0
  135. oscura/batch/logging.py +487 -0
  136. oscura/batch/metrics.py +556 -0
  137. oscura/builders/__init__.py +41 -0
  138. oscura/builders/signal_builder.py +1131 -0
  139. oscura/cli/__init__.py +14 -0
  140. oscura/cli/batch.py +339 -0
  141. oscura/cli/characterize.py +273 -0
  142. oscura/cli/compare.py +775 -0
  143. oscura/cli/decode.py +551 -0
  144. oscura/cli/main.py +247 -0
  145. oscura/cli/shell.py +350 -0
  146. oscura/comparison/__init__.py +66 -0
  147. oscura/comparison/compare.py +397 -0
  148. oscura/comparison/golden.py +487 -0
  149. oscura/comparison/limits.py +391 -0
  150. oscura/comparison/mask.py +434 -0
  151. oscura/comparison/trace_diff.py +30 -0
  152. oscura/comparison/visualization.py +481 -0
  153. oscura/compliance/__init__.py +70 -0
  154. oscura/compliance/advanced.py +756 -0
  155. oscura/compliance/masks.py +363 -0
  156. oscura/compliance/reporting.py +483 -0
  157. oscura/compliance/testing.py +298 -0
  158. oscura/component/__init__.py +38 -0
  159. oscura/component/impedance.py +365 -0
  160. oscura/component/reactive.py +598 -0
  161. oscura/component/transmission_line.py +312 -0
  162. oscura/config/__init__.py +191 -0
  163. oscura/config/defaults.py +254 -0
  164. oscura/config/loader.py +348 -0
  165. oscura/config/memory.py +271 -0
  166. oscura/config/migration.py +458 -0
  167. oscura/config/pipeline.py +1077 -0
  168. oscura/config/preferences.py +530 -0
  169. oscura/config/protocol.py +875 -0
  170. oscura/config/schema.py +713 -0
  171. oscura/config/settings.py +420 -0
  172. oscura/config/thresholds.py +599 -0
  173. oscura/convenience.py +457 -0
  174. oscura/core/__init__.py +299 -0
  175. oscura/core/audit.py +457 -0
  176. oscura/core/backend_selector.py +405 -0
  177. oscura/core/cache.py +590 -0
  178. oscura/core/cancellation.py +439 -0
  179. oscura/core/confidence.py +225 -0
  180. oscura/core/config.py +506 -0
  181. oscura/core/correlation.py +216 -0
  182. oscura/core/cross_domain.py +422 -0
  183. oscura/core/debug.py +301 -0
  184. oscura/core/edge_cases.py +541 -0
  185. oscura/core/exceptions.py +535 -0
  186. oscura/core/gpu_backend.py +523 -0
  187. oscura/core/lazy.py +832 -0
  188. oscura/core/log_query.py +540 -0
  189. oscura/core/logging.py +931 -0
  190. oscura/core/logging_advanced.py +952 -0
  191. oscura/core/memoize.py +171 -0
  192. oscura/core/memory_check.py +274 -0
  193. oscura/core/memory_guard.py +290 -0
  194. oscura/core/memory_limits.py +336 -0
  195. oscura/core/memory_monitor.py +453 -0
  196. oscura/core/memory_progress.py +465 -0
  197. oscura/core/memory_warnings.py +315 -0
  198. oscura/core/numba_backend.py +362 -0
  199. oscura/core/performance.py +352 -0
  200. oscura/core/progress.py +524 -0
  201. oscura/core/provenance.py +358 -0
  202. oscura/core/results.py +331 -0
  203. oscura/core/types.py +504 -0
  204. oscura/core/uncertainty.py +383 -0
  205. oscura/discovery/__init__.py +52 -0
  206. oscura/discovery/anomaly_detector.py +672 -0
  207. oscura/discovery/auto_decoder.py +415 -0
  208. oscura/discovery/comparison.py +497 -0
  209. oscura/discovery/quality_validator.py +528 -0
  210. oscura/discovery/signal_detector.py +769 -0
  211. oscura/dsl/__init__.py +73 -0
  212. oscura/dsl/commands.py +246 -0
  213. oscura/dsl/interpreter.py +455 -0
  214. oscura/dsl/parser.py +689 -0
  215. oscura/dsl/repl.py +172 -0
  216. oscura/exceptions.py +59 -0
  217. oscura/exploratory/__init__.py +111 -0
  218. oscura/exploratory/error_recovery.py +642 -0
  219. oscura/exploratory/fuzzy.py +513 -0
  220. oscura/exploratory/fuzzy_advanced.py +786 -0
  221. oscura/exploratory/legacy.py +831 -0
  222. oscura/exploratory/parse.py +358 -0
  223. oscura/exploratory/recovery.py +275 -0
  224. oscura/exploratory/sync.py +382 -0
  225. oscura/exploratory/unknown.py +707 -0
  226. oscura/export/__init__.py +25 -0
  227. oscura/export/wireshark/README.md +265 -0
  228. oscura/export/wireshark/__init__.py +47 -0
  229. oscura/export/wireshark/generator.py +312 -0
  230. oscura/export/wireshark/lua_builder.py +159 -0
  231. oscura/export/wireshark/templates/dissector.lua.j2 +92 -0
  232. oscura/export/wireshark/type_mapping.py +165 -0
  233. oscura/export/wireshark/validator.py +105 -0
  234. oscura/exporters/__init__.py +94 -0
  235. oscura/exporters/csv.py +303 -0
  236. oscura/exporters/exporters.py +44 -0
  237. oscura/exporters/hdf5.py +219 -0
  238. oscura/exporters/html_export.py +701 -0
  239. oscura/exporters/json_export.py +291 -0
  240. oscura/exporters/markdown_export.py +367 -0
  241. oscura/exporters/matlab_export.py +354 -0
  242. oscura/exporters/npz_export.py +219 -0
  243. oscura/exporters/spice_export.py +210 -0
  244. oscura/extensibility/__init__.py +131 -0
  245. oscura/extensibility/docs.py +752 -0
  246. oscura/extensibility/extensions.py +1125 -0
  247. oscura/extensibility/logging.py +259 -0
  248. oscura/extensibility/measurements.py +485 -0
  249. oscura/extensibility/plugins.py +414 -0
  250. oscura/extensibility/registry.py +346 -0
  251. oscura/extensibility/templates.py +913 -0
  252. oscura/extensibility/validation.py +651 -0
  253. oscura/filtering/__init__.py +89 -0
  254. oscura/filtering/base.py +563 -0
  255. oscura/filtering/convenience.py +564 -0
  256. oscura/filtering/design.py +725 -0
  257. oscura/filtering/filters.py +32 -0
  258. oscura/filtering/introspection.py +605 -0
  259. oscura/guidance/__init__.py +24 -0
  260. oscura/guidance/recommender.py +429 -0
  261. oscura/guidance/wizard.py +518 -0
  262. oscura/inference/__init__.py +251 -0
  263. oscura/inference/active_learning/README.md +153 -0
  264. oscura/inference/active_learning/__init__.py +38 -0
  265. oscura/inference/active_learning/lstar.py +257 -0
  266. oscura/inference/active_learning/observation_table.py +230 -0
  267. oscura/inference/active_learning/oracle.py +78 -0
  268. oscura/inference/active_learning/teachers/__init__.py +15 -0
  269. oscura/inference/active_learning/teachers/simulator.py +192 -0
  270. oscura/inference/adaptive_tuning.py +453 -0
  271. oscura/inference/alignment.py +653 -0
  272. oscura/inference/bayesian.py +943 -0
  273. oscura/inference/binary.py +1016 -0
  274. oscura/inference/crc_reverse.py +711 -0
  275. oscura/inference/logic.py +288 -0
  276. oscura/inference/message_format.py +1305 -0
  277. oscura/inference/protocol.py +417 -0
  278. oscura/inference/protocol_dsl.py +1084 -0
  279. oscura/inference/protocol_library.py +1230 -0
  280. oscura/inference/sequences.py +809 -0
  281. oscura/inference/signal_intelligence.py +1509 -0
  282. oscura/inference/spectral.py +215 -0
  283. oscura/inference/state_machine.py +634 -0
  284. oscura/inference/stream.py +918 -0
  285. oscura/integrations/__init__.py +59 -0
  286. oscura/integrations/llm.py +1827 -0
  287. oscura/jupyter/__init__.py +32 -0
  288. oscura/jupyter/display.py +268 -0
  289. oscura/jupyter/magic.py +334 -0
  290. oscura/loaders/__init__.py +526 -0
  291. oscura/loaders/binary.py +69 -0
  292. oscura/loaders/configurable.py +1255 -0
  293. oscura/loaders/csv.py +26 -0
  294. oscura/loaders/csv_loader.py +473 -0
  295. oscura/loaders/hdf5.py +9 -0
  296. oscura/loaders/hdf5_loader.py +510 -0
  297. oscura/loaders/lazy.py +370 -0
  298. oscura/loaders/mmap_loader.py +583 -0
  299. oscura/loaders/numpy_loader.py +436 -0
  300. oscura/loaders/pcap.py +432 -0
  301. oscura/loaders/preprocessing.py +368 -0
  302. oscura/loaders/rigol.py +287 -0
  303. oscura/loaders/sigrok.py +321 -0
  304. oscura/loaders/tdms.py +367 -0
  305. oscura/loaders/tektronix.py +711 -0
  306. oscura/loaders/validation.py +584 -0
  307. oscura/loaders/vcd.py +464 -0
  308. oscura/loaders/wav.py +233 -0
  309. oscura/math/__init__.py +45 -0
  310. oscura/math/arithmetic.py +824 -0
  311. oscura/math/interpolation.py +413 -0
  312. oscura/onboarding/__init__.py +39 -0
  313. oscura/onboarding/help.py +498 -0
  314. oscura/onboarding/tutorials.py +405 -0
  315. oscura/onboarding/wizard.py +466 -0
  316. oscura/optimization/__init__.py +19 -0
  317. oscura/optimization/parallel.py +440 -0
  318. oscura/optimization/search.py +532 -0
  319. oscura/pipeline/__init__.py +43 -0
  320. oscura/pipeline/base.py +338 -0
  321. oscura/pipeline/composition.py +242 -0
  322. oscura/pipeline/parallel.py +448 -0
  323. oscura/pipeline/pipeline.py +375 -0
  324. oscura/pipeline/reverse_engineering.py +1119 -0
  325. oscura/plugins/__init__.py +122 -0
  326. oscura/plugins/base.py +272 -0
  327. oscura/plugins/cli.py +497 -0
  328. oscura/plugins/discovery.py +411 -0
  329. oscura/plugins/isolation.py +418 -0
  330. oscura/plugins/lifecycle.py +959 -0
  331. oscura/plugins/manager.py +493 -0
  332. oscura/plugins/registry.py +421 -0
  333. oscura/plugins/versioning.py +372 -0
  334. oscura/py.typed +0 -0
  335. oscura/quality/__init__.py +65 -0
  336. oscura/quality/ensemble.py +740 -0
  337. oscura/quality/explainer.py +338 -0
  338. oscura/quality/scoring.py +616 -0
  339. oscura/quality/warnings.py +456 -0
  340. oscura/reporting/__init__.py +248 -0
  341. oscura/reporting/advanced.py +1234 -0
  342. oscura/reporting/analyze.py +448 -0
  343. oscura/reporting/argument_preparer.py +596 -0
  344. oscura/reporting/auto_report.py +507 -0
  345. oscura/reporting/batch.py +615 -0
  346. oscura/reporting/chart_selection.py +223 -0
  347. oscura/reporting/comparison.py +330 -0
  348. oscura/reporting/config.py +615 -0
  349. oscura/reporting/content/__init__.py +39 -0
  350. oscura/reporting/content/executive.py +127 -0
  351. oscura/reporting/content/filtering.py +191 -0
  352. oscura/reporting/content/minimal.py +257 -0
  353. oscura/reporting/content/verbosity.py +162 -0
  354. oscura/reporting/core.py +508 -0
  355. oscura/reporting/core_formats/__init__.py +17 -0
  356. oscura/reporting/core_formats/multi_format.py +210 -0
  357. oscura/reporting/engine.py +836 -0
  358. oscura/reporting/export.py +366 -0
  359. oscura/reporting/formatting/__init__.py +129 -0
  360. oscura/reporting/formatting/emphasis.py +81 -0
  361. oscura/reporting/formatting/numbers.py +403 -0
  362. oscura/reporting/formatting/standards.py +55 -0
  363. oscura/reporting/formatting.py +466 -0
  364. oscura/reporting/html.py +578 -0
  365. oscura/reporting/index.py +590 -0
  366. oscura/reporting/multichannel.py +296 -0
  367. oscura/reporting/output.py +379 -0
  368. oscura/reporting/pdf.py +373 -0
  369. oscura/reporting/plots.py +731 -0
  370. oscura/reporting/pptx_export.py +360 -0
  371. oscura/reporting/renderers/__init__.py +11 -0
  372. oscura/reporting/renderers/pdf.py +94 -0
  373. oscura/reporting/sections.py +471 -0
  374. oscura/reporting/standards.py +680 -0
  375. oscura/reporting/summary_generator.py +368 -0
  376. oscura/reporting/tables.py +397 -0
  377. oscura/reporting/template_system.py +724 -0
  378. oscura/reporting/templates/__init__.py +15 -0
  379. oscura/reporting/templates/definition.py +205 -0
  380. oscura/reporting/templates/index.html +649 -0
  381. oscura/reporting/templates/index.md +173 -0
  382. oscura/schemas/__init__.py +158 -0
  383. oscura/schemas/bus_configuration.json +322 -0
  384. oscura/schemas/device_mapping.json +182 -0
  385. oscura/schemas/packet_format.json +418 -0
  386. oscura/schemas/protocol_definition.json +363 -0
  387. oscura/search/__init__.py +16 -0
  388. oscura/search/anomaly.py +292 -0
  389. oscura/search/context.py +149 -0
  390. oscura/search/pattern.py +160 -0
  391. oscura/session/__init__.py +34 -0
  392. oscura/session/annotations.py +289 -0
  393. oscura/session/history.py +313 -0
  394. oscura/session/session.py +445 -0
  395. oscura/streaming/__init__.py +43 -0
  396. oscura/streaming/chunked.py +611 -0
  397. oscura/streaming/progressive.py +393 -0
  398. oscura/streaming/realtime.py +622 -0
  399. oscura/testing/__init__.py +54 -0
  400. oscura/testing/synthetic.py +808 -0
  401. oscura/triggering/__init__.py +68 -0
  402. oscura/triggering/base.py +229 -0
  403. oscura/triggering/edge.py +353 -0
  404. oscura/triggering/pattern.py +344 -0
  405. oscura/triggering/pulse.py +581 -0
  406. oscura/triggering/window.py +453 -0
  407. oscura/ui/__init__.py +48 -0
  408. oscura/ui/formatters.py +526 -0
  409. oscura/ui/progressive_display.py +340 -0
  410. oscura/utils/__init__.py +99 -0
  411. oscura/utils/autodetect.py +338 -0
  412. oscura/utils/buffer.py +389 -0
  413. oscura/utils/lazy.py +407 -0
  414. oscura/utils/lazy_imports.py +147 -0
  415. oscura/utils/memory.py +836 -0
  416. oscura/utils/memory_advanced.py +1326 -0
  417. oscura/utils/memory_extensions.py +465 -0
  418. oscura/utils/progressive.py +352 -0
  419. oscura/utils/windowing.py +362 -0
  420. oscura/visualization/__init__.py +321 -0
  421. oscura/visualization/accessibility.py +526 -0
  422. oscura/visualization/annotations.py +374 -0
  423. oscura/visualization/axis_scaling.py +305 -0
  424. oscura/visualization/colors.py +453 -0
  425. oscura/visualization/digital.py +337 -0
  426. oscura/visualization/eye.py +420 -0
  427. oscura/visualization/histogram.py +281 -0
  428. oscura/visualization/interactive.py +858 -0
  429. oscura/visualization/jitter.py +702 -0
  430. oscura/visualization/keyboard.py +394 -0
  431. oscura/visualization/layout.py +365 -0
  432. oscura/visualization/optimization.py +1028 -0
  433. oscura/visualization/palettes.py +446 -0
  434. oscura/visualization/plot.py +92 -0
  435. oscura/visualization/power.py +290 -0
  436. oscura/visualization/power_extended.py +626 -0
  437. oscura/visualization/presets.py +467 -0
  438. oscura/visualization/protocols.py +932 -0
  439. oscura/visualization/render.py +207 -0
  440. oscura/visualization/rendering.py +444 -0
  441. oscura/visualization/reverse_engineering.py +791 -0
  442. oscura/visualization/signal_integrity.py +808 -0
  443. oscura/visualization/specialized.py +553 -0
  444. oscura/visualization/spectral.py +811 -0
  445. oscura/visualization/styles.py +381 -0
  446. oscura/visualization/thumbnails.py +311 -0
  447. oscura/visualization/time_axis.py +351 -0
  448. oscura/visualization/waveform.py +367 -0
  449. oscura/workflow/__init__.py +13 -0
  450. oscura/workflow/dag.py +377 -0
  451. oscura/workflows/__init__.py +58 -0
  452. oscura/workflows/compliance.py +280 -0
  453. oscura/workflows/digital.py +272 -0
  454. oscura/workflows/multi_trace.py +502 -0
  455. oscura/workflows/power.py +178 -0
  456. oscura/workflows/protocol.py +492 -0
  457. oscura/workflows/reverse_engineering.py +639 -0
  458. oscura/workflows/signal_integrity.py +227 -0
  459. oscura-0.1.1.dist-info/METADATA +300 -0
  460. oscura-0.1.1.dist-info/RECORD +463 -0
  461. oscura-0.1.1.dist-info/entry_points.txt +2 -0
  462. {oscura-0.0.1.dist-info → oscura-0.1.1.dist-info}/licenses/LICENSE +1 -1
  463. oscura-0.0.1.dist-info/METADATA +0 -63
  464. oscura-0.0.1.dist-info/RECORD +0 -5
  465. {oscura-0.0.1.dist-info → oscura-0.1.1.dist-info}/WHEEL +0 -0
@@ -0,0 +1,300 @@
1
+ """Result aggregation for batch analysis.
2
+
3
+
4
+ This module provides statistical aggregation and reporting for batch
5
+ analysis results, including outlier detection and export capabilities.
6
+ """
7
+
8
+ from pathlib import Path
9
+ from typing import Any
10
+
11
+ import numpy as np
12
+ import pandas as pd
13
+
14
+
15
+ def aggregate_results(
16
+ results: pd.DataFrame,
17
+ *,
18
+ metrics: list[str] | None = None,
19
+ outlier_threshold: float = 3.0,
20
+ include_plots: bool = False,
21
+ output_format: str = "dict",
22
+ output_file: str | Path | None = None,
23
+ ) -> dict[str, Any] | pd.DataFrame:
24
+ """Aggregate results from batch analysis into summary statistics.
25
+
26
+ : Computes comprehensive statistics (mean, std, min, max,
27
+ outliers) for each metric in the batch results. Supports export to various
28
+ formats and optional visualization generation.
29
+
30
+ Args:
31
+ results: DataFrame from batch_analyze() containing analysis results
32
+ metrics: List of column names to aggregate (default: all numeric columns)
33
+ outlier_threshold: Z-score threshold for outlier detection (default: 3.0)
34
+ include_plots: Generate comparison plots across files (default: False)
35
+ output_format: Output format - 'dict', 'dataframe', 'csv', 'excel', 'html'
36
+ output_file: Optional output file path for export formats
37
+
38
+ Returns:
39
+ Dictionary or DataFrame with summary statistics:
40
+ - count: Number of valid values
41
+ - mean: Mean value
42
+ - std: Standard deviation
43
+ - min: Minimum value
44
+ - max: Maximum value
45
+ - median: Median value
46
+ - q25: 25th percentile
47
+ - q75: 75th percentile
48
+ - outliers: List of outlier values
49
+ - outlier_files: List of files containing outliers
50
+
51
+ Raises:
52
+ ValueError: If no numeric metrics are found in results.
53
+
54
+ Examples:
55
+ >>> results = osc.batch_analyze(files, osc.characterize_buffer)
56
+ >>> summary = osc.aggregate_results(
57
+ ... results,
58
+ ... metrics=['rise_time', 'fall_time'],
59
+ ... outlier_threshold=2.5
60
+ ... )
61
+ >>> print(summary['rise_time']['mean'])
62
+ >>> print(summary['rise_time']['outlier_files'])
63
+
64
+ Notes:
65
+ - Outliers detected using IQR method: values outside [Q1 - k*IQR, Q3 + k*IQR]
66
+ where k = (threshold / 3.0) * 1.5 (more robust than z-score for heavy-tailed data)
67
+ - Non-numeric columns are automatically skipped
68
+ - Missing values (NaN) are excluded from statistics
69
+ - CSV/Excel/HTML export requires output_file parameter
70
+
71
+ References:
72
+ BATCH-002: Result Aggregation
73
+ """
74
+ if results.empty:
75
+ return {} if output_format == "dict" else pd.DataFrame()
76
+
77
+ # Determine metrics to analyze
78
+ if metrics is None:
79
+ # Auto-select all numeric columns except 'file' and 'error'
80
+ metrics = results.select_dtypes(include=[np.number]).columns.tolist()
81
+ metrics = [m for m in metrics if m not in ["file", "error"]]
82
+
83
+ if not metrics:
84
+ raise ValueError("No numeric metrics found in results")
85
+
86
+ # Compute aggregated statistics
87
+ aggregated: dict[str, dict[str, Any]] = {}
88
+
89
+ for metric in metrics:
90
+ if metric not in results.columns:
91
+ continue
92
+
93
+ # Extract valid (non-null) values
94
+ values = results[metric].dropna()
95
+
96
+ if values.empty:
97
+ aggregated[metric] = {
98
+ "count": 0,
99
+ "mean": np.nan,
100
+ "std": np.nan,
101
+ "min": np.nan,
102
+ "max": np.nan,
103
+ "median": np.nan,
104
+ "q25": np.nan,
105
+ "q75": np.nan,
106
+ "outliers": [],
107
+ "outlier_files": [],
108
+ }
109
+ continue
110
+
111
+ # Basic statistics
112
+ stats = {
113
+ "count": len(values),
114
+ "mean": float(values.mean()),
115
+ "std": float(values.std()),
116
+ "min": float(values.min()),
117
+ "max": float(values.max()),
118
+ "median": float(values.median()),
119
+ "q25": float(values.quantile(0.25)),
120
+ "q75": float(values.quantile(0.75)),
121
+ }
122
+
123
+ # Outlier detection using IQR method (more robust than z-score)
124
+ # IQR method: outliers are values outside [Q1 - k*IQR, Q3 + k*IQR]
125
+ # where k = outlier_threshold * 1.5 (standard is k=1.5, we scale by threshold)
126
+ if len(values) > 3: # Need at least 4 values for meaningful IQR
127
+ q1 = stats["q25"]
128
+ q3 = stats["q75"]
129
+ iqr = q3 - q1
130
+
131
+ # Scale IQR multiplier by threshold (default 3.0 -> 2.0 * 1.5 = 3.0)
132
+ k = (outlier_threshold / 3.0) * 1.5
133
+
134
+ lower_bound = q1 - k * iqr
135
+ upper_bound = q3 + k * iqr
136
+
137
+ outlier_mask = (values < lower_bound) | (values > upper_bound)
138
+ outlier_indices = values[outlier_mask].index.tolist()
139
+ stats["outliers"] = values[outlier_mask].tolist()
140
+
141
+ # Get corresponding filenames if available
142
+ if "file" in results.columns:
143
+ stats["outlier_files"] = results.loc[outlier_indices, "file"].tolist()
144
+ else:
145
+ stats["outlier_files"] = outlier_indices
146
+ else:
147
+ stats["outliers"] = [] # type: ignore[assignment]
148
+ stats["outlier_files"] = [] # type: ignore[assignment]
149
+
150
+ aggregated[metric] = stats
151
+
152
+ # Generate plots if requested
153
+ if include_plots:
154
+ # Import here to avoid circular dependency
155
+ try:
156
+ import matplotlib.pyplot as plt
157
+
158
+ for metric in metrics:
159
+ if metric not in aggregated:
160
+ continue
161
+
162
+ _fig, (ax1, ax2) = plt.subplots(1, 2, figsize=(12, 4))
163
+
164
+ # Histogram
165
+ results[metric].dropna().hist(ax=ax1, bins=30)
166
+ ax1.axvline(
167
+ aggregated[metric]["mean"],
168
+ color="r",
169
+ linestyle="--",
170
+ label="Mean",
171
+ )
172
+ ax1.axvline(
173
+ aggregated[metric]["median"],
174
+ color="g",
175
+ linestyle="--",
176
+ label="Median",
177
+ )
178
+ ax1.set_xlabel(metric)
179
+ ax1.set_ylabel("Count")
180
+ ax1.legend()
181
+ ax1.set_title(f"{metric} Distribution")
182
+
183
+ # Box plot
184
+ ax2.boxplot(results[metric].dropna())
185
+ ax2.set_ylabel(metric)
186
+ ax2.set_title(f"{metric} Box Plot")
187
+
188
+ plt.tight_layout()
189
+
190
+ # Save or show based on output_file
191
+ if output_file:
192
+ plot_file = Path(output_file).with_suffix("") / f"{metric}_plot.png"
193
+ plot_file.parent.mkdir(parents=True, exist_ok=True)
194
+ plt.savefig(plot_file)
195
+ else:
196
+ plt.show()
197
+
198
+ plt.close()
199
+
200
+ except ImportError:
201
+ pass # Silently skip plotting if matplotlib not available
202
+
203
+ # Format output
204
+ if output_format == "dict":
205
+ return aggregated
206
+
207
+ elif output_format == "dataframe":
208
+ # Convert to DataFrame with metrics as rows
209
+ df = pd.DataFrame(aggregated).T
210
+ # Drop list columns for DataFrame format
211
+ df = df.drop(columns=["outliers", "outlier_files"], errors="ignore")
212
+ return df
213
+
214
+ elif output_format in ["csv", "excel", "html"]:
215
+ if not output_file:
216
+ raise ValueError(f"{output_format} format requires output_file parameter")
217
+
218
+ df = pd.DataFrame(aggregated).T
219
+ df = df.drop(columns=["outliers", "outlier_files"], errors="ignore")
220
+
221
+ if output_format == "csv":
222
+ df.to_csv(output_file)
223
+ elif output_format == "excel":
224
+ df.to_excel(output_file)
225
+ elif output_format == "html":
226
+ # Generate HTML report
227
+ html = _generate_html_report(results, aggregated, metrics)
228
+ Path(output_file).write_text(html)
229
+
230
+ return df
231
+
232
+ else:
233
+ raise ValueError(f"Unknown output_format: {output_format}")
234
+
235
+
236
+ def _generate_html_report(
237
+ results: pd.DataFrame,
238
+ aggregated: dict[str, dict[str, Any]],
239
+ metrics: list[str],
240
+ ) -> str:
241
+ """Generate HTML report for batch analysis results."""
242
+ html = """
243
+ <!DOCTYPE html>
244
+ <html>
245
+ <head>
246
+ <title>Batch Analysis Report</title>
247
+ <style>
248
+ body { font-family: Arial, sans-serif; margin: 20px; }
249
+ h1 { color: #333; }
250
+ h2 { color: #666; margin-top: 30px; }
251
+ table { border-collapse: collapse; width: 100%; margin: 20px 0; }
252
+ th, td { border: 1px solid #ddd; padding: 8px; text-align: left; }
253
+ th { background-color: #4CAF50; color: white; }
254
+ tr:nth-child(even) { background-color: #f2f2f2; }
255
+ .outlier { background-color: #ffcccc; }
256
+ </style>
257
+ </head>
258
+ <body>
259
+ <h1>Batch Analysis Report</h1>
260
+ """
261
+ # Summary statistics table
262
+ html += "<h2>Summary Statistics</h2>\n<table>\n"
263
+ html += "<tr><th>Metric</th><th>Count</th><th>Mean</th><th>Std</th>"
264
+ html += "<th>Min</th><th>Median</th><th>Max</th><th>Outliers</th></tr>\n"
265
+
266
+ for metric in metrics:
267
+ if metric not in aggregated:
268
+ continue
269
+ stats = aggregated[metric]
270
+ html += "<tr>"
271
+ html += f"<td>{metric}</td>"
272
+ html += f"<td>{stats['count']}</td>"
273
+ html += f"<td>{stats['mean']:.4g}</td>"
274
+ html += f"<td>{stats['std']:.4g}</td>"
275
+ html += f"<td>{stats['min']:.4g}</td>"
276
+ html += f"<td>{stats['median']:.4g}</td>"
277
+ html += f"<td>{stats['max']:.4g}</td>"
278
+ html += f"<td>{len(stats['outliers'])}</td>"
279
+ html += "</tr>\n"
280
+
281
+ html += "</table>\n"
282
+
283
+ # Outlier details
284
+ has_outliers = any(len(aggregated[m]["outliers"]) > 0 for m in metrics if m in aggregated)
285
+
286
+ if has_outliers:
287
+ html += "<h2>Outliers Detected</h2>\n"
288
+ for metric in metrics:
289
+ if metric not in aggregated:
290
+ continue
291
+ stats = aggregated[metric]
292
+ if stats["outliers"]:
293
+ html += f"<h3>{metric}</h3>\n<table>\n"
294
+ html += "<tr><th>File</th><th>Value</th></tr>\n"
295
+ for file, value in zip(stats["outlier_files"], stats["outliers"], strict=False):
296
+ html += f"<tr class='outlier'><td>{file}</td><td>{value:.4g}</td></tr>\n"
297
+ html += "</table>\n"
298
+
299
+ html += "</body>\n</html>"
300
+ return html
@@ -0,0 +1,139 @@
1
+ """Multi-file batch analysis with parallel execution support.
2
+
3
+
4
+ This module provides parallel batch processing of signal files using
5
+ concurrent.futures for efficient multi-core utilization.
6
+ """
7
+
8
+ from collections.abc import Callable
9
+ from concurrent.futures import ProcessPoolExecutor, ThreadPoolExecutor, as_completed
10
+ from pathlib import Path
11
+ from typing import Any
12
+
13
+ import pandas as pd
14
+
15
+
16
+ def batch_analyze(
17
+ files: list[str | Path],
18
+ analysis_fn: Callable[[str | Path], dict[str, Any]],
19
+ *,
20
+ parallel: bool = False,
21
+ workers: int | None = None,
22
+ progress_callback: Callable[[int, int, str], None] | None = None,
23
+ use_threads: bool = False,
24
+ **config: Any,
25
+ ) -> pd.DataFrame:
26
+ """Analyze multiple files with the same analysis configuration.
27
+
28
+ : Multi-file analysis with parallel execution support
29
+ via concurrent.futures. Returns aggregated results as a DataFrame for
30
+ easy statistical analysis and export.
31
+
32
+ Args:
33
+ files: List of file paths to analyze
34
+ analysis_fn: Analysis function to apply to each file.
35
+ Must accept a file path and return a dict of results.
36
+ parallel: Enable parallel processing (default: False)
37
+ workers: Number of parallel workers (default: CPU count)
38
+ progress_callback: Optional callback for progress updates.
39
+ Called with (current, total, filename) after each file.
40
+ use_threads: Use ThreadPoolExecutor instead of ProcessPoolExecutor
41
+ (useful for I/O-bound tasks, default: False)
42
+ **config: Additional keyword arguments passed to analysis_fn
43
+
44
+ Returns:
45
+ DataFrame with one row per file, columns from analysis results.
46
+ Always includes a 'file' column with the input filename.
47
+
48
+ Examples:
49
+ >>> import oscura as osc
50
+ >>> import glob
51
+ >>> files = glob.glob('captures/*.wfm')
52
+ >>> results = osc.batch_analyze(
53
+ ... files,
54
+ ... analysis_fn=osc.characterize_buffer,
55
+ ... parallel=True,
56
+ ... workers=4
57
+ ... )
58
+ >>> print(results[['file', 'rise_time', 'fall_time', 'status']])
59
+ >>> results.to_csv('batch_results.csv')
60
+
61
+ Notes:
62
+ - Use parallel=True for CPU-bound analysis functions
63
+ - Use use_threads=True for I/O-bound operations (file loading)
64
+ - Progress callback is called from worker threads/processes
65
+ - All exceptions during analysis are caught and stored in 'error' column
66
+
67
+ References:
68
+ BATCH-001: Multi-File Analysis
69
+ """
70
+ if not files:
71
+ return pd.DataFrame()
72
+
73
+ # Wrapper to include config in analysis calls
74
+ def _wrapped_analysis(filepath: str | Path) -> dict[str, Any]:
75
+ try:
76
+ result = analysis_fn(filepath, **config)
77
+ # Ensure result is a dict
78
+ if not isinstance(result, dict):
79
+ result = {"result": result} # type: ignore[unreachable]
80
+ result["file"] = str(filepath)
81
+ result["error"] = None
82
+ return result
83
+ except Exception as e:
84
+ # Return error info on failure
85
+ return {
86
+ "file": str(filepath),
87
+ "error": str(e),
88
+ }
89
+
90
+ results: list[dict[str, Any]] = []
91
+ total = len(files)
92
+
93
+ if parallel:
94
+ # Use concurrent.futures for parallel execution
95
+ executor_class = ThreadPoolExecutor if use_threads else ProcessPoolExecutor
96
+ with executor_class(max_workers=workers) as executor:
97
+ # Submit all tasks
98
+ future_to_file = {executor.submit(_wrapped_analysis, f): f for f in files}
99
+
100
+ # Process results as they complete
101
+ for i, future in enumerate(as_completed(future_to_file), 1):
102
+ filepath = future_to_file[future]
103
+ try:
104
+ result = future.result()
105
+ results.append(result)
106
+
107
+ if progress_callback:
108
+ progress_callback(i, total, str(filepath))
109
+ except Exception as e:
110
+ # Catch execution errors
111
+ results.append(
112
+ {
113
+ "file": str(filepath),
114
+ "error": f"Execution error: {e}",
115
+ }
116
+ )
117
+
118
+ else:
119
+ # Sequential processing
120
+ for i, filepath in enumerate(files, 1):
121
+ result = _wrapped_analysis(filepath)
122
+ results.append(result)
123
+
124
+ if progress_callback:
125
+ progress_callback(i, total, str(filepath))
126
+
127
+ # Convert to DataFrame
128
+ df = pd.DataFrame(results)
129
+
130
+ # Reorder columns: file first, error last
131
+ cols = df.columns.tolist()
132
+ if "file" in cols:
133
+ cols.remove("file")
134
+ cols = ["file", *cols]
135
+ if "error" in cols:
136
+ cols.remove("error")
137
+ cols = [*cols, "error"]
138
+
139
+ return df[cols]