evolver-tools 22.0.0__tar.gz → 24.0.0__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- {evolver_tools-22.0.0/src/evolver_tools.egg-info → evolver_tools-24.0.0}/PKG-INFO +2 -2
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/pyproject.toml +2 -2
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/cli.py +1 -1
- evolver_tools-24.0.0/src/evolver_tools/vendor/base58.py +93 -0
- evolver_tools-24.0.0/src/evolver_tools/vendor/file_find.py +91 -0
- evolver_tools-24.0.0/src/evolver_tools/vendor/gzip_cli.py +81 -0
- evolver_tools-24.0.0/src/evolver_tools/vendor/ip_info.py +70 -0
- evolver_tools-24.0.0/src/evolver_tools/vendor/json_diff.py +96 -0
- evolver_tools-24.0.0/src/evolver_tools/vendor/json_merge.py +104 -0
- evolver_tools-24.0.0/src/evolver_tools/vendor/jwt_decode.py +81 -0
- evolver_tools-24.0.0/src/evolver_tools/vendor/markdown_lint.py +114 -0
- evolver_tools-24.0.0/src/evolver_tools/vendor/tsv2csv.py +73 -0
- evolver_tools-24.0.0/src/evolver_tools/vendor/world_clock.py +80 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0/src/evolver_tools.egg-info}/PKG-INFO +2 -2
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools.egg-info/SOURCES.txt +10 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/LICENSE +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/README.md +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/setup.cfg +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/autoreg.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/agent_b_tool.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ansi_strip.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/api_tester.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ascii_gen.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/audit_log.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/b64/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/b64/b64.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/backup.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/banner/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/banner/banner.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/banner.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/base32.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/bookmark.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/cal_tool/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/cal_tool/cli.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/calendar_cli.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/cert_check.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/cert_info.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/changelog_gen/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/changelog_gen/changelog_gen.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/changelog_gen.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/chart_cli/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/chart_cli/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/checksum_dir.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/clipboard/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/clipboard/clipboard.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/code_auditor.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/code_stats.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/colorize.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/colors/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/colors/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/config_validator.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/config_vault.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/cowsay.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/crc_check.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/cron/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/cron/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/cron_pretty.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/crontab_helper.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/crypto_box.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/crypto_price.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/csv2json.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/csv_merge.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/csv_stats/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/csv_stats/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/csv_stats/analyzer.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/csv_stats/cli.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/csv_validate.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/csv_view.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/date_diff.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/db_mate.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/db_schema.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/dep_graph.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/dev_dashboard.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/dice_roll.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/diff_csv.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/diff_files.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/diff_tool/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/diff_tool/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/dirsize/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/disk_cleanup.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/disk_usage/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/disk_usage/disk_usage.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/dns_lookup.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/docker_helper.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/dt_convert.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/env_diff.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/env_manager.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/env_template.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/envcheck/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/epoch.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/excel2csv.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/factor.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ff/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ff/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/figlet_cli.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/figlet_tool.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/file_encrypt.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/file_joiner.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/file_splitter.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/file_watch.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/find_dups/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/find_dups/cli.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/firewall_rule.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/fmt/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/fmt/fmt.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/fold.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/geo_ip.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/git_branch_cleaner.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/git_log_pretty.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/git_stats.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/hash_check.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/hash_file.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/hashsum/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/hashsum/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/hex_tool.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/hexdump.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/html2markdown.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/html2md.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/http_headers.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/http_live/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/http_live/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/http_server.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/image_meta.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ini2json.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ini_parser/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ini_parser/ini_parser.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ip_location.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ipcalc/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ipcalc/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ipinfo/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ipinfo/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/join.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/joke.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/jq_lite/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/jq_lite/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/json2csv/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/json2csv/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/json2ini.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/json_flatten.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/json_pretty/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/json_pretty/json_pretty.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/json_schema_validate.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/jsonql/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/jsonql/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/key_value_store.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/license.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/license_cli/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/license_cli/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/license_cli/cli.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/log_analyzer.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/log_hawk.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/log_tail.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/macrogen.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/markdown_check/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/markdown_preview.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/markdown_toc.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/math_eval.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/media_studio.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/morse.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/nb/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/nb/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/net_analyzer.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/net_speed.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/network_scan.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/nl.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/note_taker.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/otp_gen.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/passgen/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/password_strength.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/pdf_info.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/pdf_text.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/pipe_viewer.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/pomodoro.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/port_scan.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/portcheck/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/portcheck/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/pr_tool/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/pr_tool/pr_tool.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/process_kill.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/progress_bar.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/project_doctor/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/project_doctor/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/qc_calc.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/qc_report.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/qc_sample.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/qr_cli.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/qrcode.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/quote.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/quote_tool/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/quote_tool/quote.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/rainbow.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/random.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/reminder.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ren/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ren/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/replace_text.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/restore.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/rot13.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/route_trace.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/scan_ports.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/screen_recorder.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/screenshot_cli.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/search_files.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/search_history.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/secret_scanner.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/seq.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/service_check.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/shuffle.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/siege_lite/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/siege_lite/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/smellfinder/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/smellfinder/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/sort/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/sort/sort.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/spinner.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/split.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/split_tool/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/split_tool/split.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/sql2csv.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/sqlite_cli/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/sqlite_cli/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ssh_key_gen.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/ssl_check.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/stopwatch.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/subnet.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/sysmon/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/sysmon/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/sysmon_pro.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/system_info.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/temp_cleaner.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/template.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/text_stats.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/timeout.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/timer/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/timer_pro/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/timer_pro/timer_pro.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/timer_pro.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/todo_cli.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/toml2json.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/tr.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/tree.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/treedir/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/treedir/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/uniq_tool/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/uniq_tool/uniq.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/unit_convert.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/uri_encode.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/url_parser.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/urlparse_tool/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/urlparse_tool/cli.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/uuid_gen.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/uuid_tool/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/uuid_tool/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/watch.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/weather_cli.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/web_download.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/web_summary/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/web_summary/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/whois_lookup.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/wordcount/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/wordcount/__main__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/xml2json.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/yaml2json/__init__.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/yaml2json/yaml2json.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/yaml2toml.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/yaml_validate.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools/vendor/yes.py +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools.egg-info/dependency_links.txt +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools.egg-info/entry_points.txt +0 -0
- {evolver_tools-22.0.0 → evolver_tools-24.0.0}/src/evolver_tools.egg-info/top_level.txt +0 -0
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
Metadata-Version: 2.4
|
|
2
2
|
Name: evolver-tools
|
|
3
|
-
Version:
|
|
4
|
-
Summary:
|
|
3
|
+
Version: 24.0.0
|
|
4
|
+
Summary: 201 CLI tools + 9 flagship projects — one pip install
|
|
5
5
|
Author: EVOLVER
|
|
6
6
|
License-Expression: MIT
|
|
7
7
|
Project-URL: Homepage, https://evolver-dev.github.io/evolver-tools
|
|
@@ -4,8 +4,8 @@ build-backend = "setuptools.build_meta"
|
|
|
4
4
|
|
|
5
5
|
[project]
|
|
6
6
|
name = "evolver-tools"
|
|
7
|
-
version = "
|
|
8
|
-
description = "
|
|
7
|
+
version = "24.0.0"
|
|
8
|
+
description = "201 CLI tools + 9 flagship projects — one pip install"
|
|
9
9
|
readme = "README.md"
|
|
10
10
|
license = "MIT"
|
|
11
11
|
requires-python = ">=3.8"
|
|
@@ -14,7 +14,7 @@ from evolver_tools.autoreg import auto_discover
|
|
|
14
14
|
def list_tools():
|
|
15
15
|
"""Display all available tools."""
|
|
16
16
|
tools = auto_discover()
|
|
17
|
-
print(f'\x1b[1;36m===== EVOLVER Tools
|
|
17
|
+
print(f'\x1b[1;36m===== EVOLVER Tools v23.0.0 =====\x1b[0m')
|
|
18
18
|
print()
|
|
19
19
|
for name, info in sorted(tools.items()):
|
|
20
20
|
print(f' \033[1;33m{name:<18}\033[0m {info["desc"]}')
|
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""base58 — Base58 encode/decode (Bitcoin-style alphabet).
|
|
3
|
+
|
|
4
|
+
Usage: base58 encode hello
|
|
5
|
+
base58 decode 3yZe7d
|
|
6
|
+
echo hello | base58 encode
|
|
7
|
+
|
|
8
|
+
Base58 uses Bitcoin alphabet (123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz).
|
|
9
|
+
No '0', 'O', 'I', 'l' — safe for human transcription.
|
|
10
|
+
"""
|
|
11
|
+
|
|
12
|
+
import sys
|
|
13
|
+
|
|
14
|
+
TOOL_META = {
|
|
15
|
+
"name": "base58",
|
|
16
|
+
"func": "main",
|
|
17
|
+
"desc": "Base58 encode/decode (Bitcoin-style alphabet)",
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
ALPHABET = b'123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'
|
|
21
|
+
BASE = 58
|
|
22
|
+
|
|
23
|
+
def b58encode(data):
|
|
24
|
+
if isinstance(data, str):
|
|
25
|
+
data = data.encode()
|
|
26
|
+
n = int.from_bytes(data, 'big')
|
|
27
|
+
if n == 0:
|
|
28
|
+
return '1'
|
|
29
|
+
res = []
|
|
30
|
+
while n > 0:
|
|
31
|
+
n, r = divmod(n, BASE)
|
|
32
|
+
res.append(chr(ALPHABET[r]))
|
|
33
|
+
# Leading zeros
|
|
34
|
+
leading = 0
|
|
35
|
+
for b in data:
|
|
36
|
+
if b == 0:
|
|
37
|
+
leading += 1
|
|
38
|
+
else:
|
|
39
|
+
break
|
|
40
|
+
return '1' * leading + ''.join(reversed(res))
|
|
41
|
+
|
|
42
|
+
|
|
43
|
+
def b58decode(s):
|
|
44
|
+
n = 0
|
|
45
|
+
for c in s:
|
|
46
|
+
idx = ALPHABET.find(c.encode() if isinstance(c, str) else c)
|
|
47
|
+
if idx == -1:
|
|
48
|
+
raise ValueError(f"Invalid Base58 character: {c}")
|
|
49
|
+
n = n * BASE + idx
|
|
50
|
+
# Count leading '1's
|
|
51
|
+
leading = 0
|
|
52
|
+
for c in s:
|
|
53
|
+
if c == '1':
|
|
54
|
+
leading += 1
|
|
55
|
+
else:
|
|
56
|
+
break
|
|
57
|
+
result = n.to_bytes((n.bit_length() + 7) // 8, 'big') if n > 0 else b''
|
|
58
|
+
return b'\x00' * leading + result
|
|
59
|
+
|
|
60
|
+
|
|
61
|
+
def main():
|
|
62
|
+
args = sys.argv[1:]
|
|
63
|
+
if not args or args[0] in ('-h', '--help'):
|
|
64
|
+
print(__doc__)
|
|
65
|
+
return
|
|
66
|
+
|
|
67
|
+
mode = args[0]
|
|
68
|
+
if mode not in ('encode', 'decode'):
|
|
69
|
+
print("Usage: base58 encode|decode [text]", file=sys.stderr)
|
|
70
|
+
sys.exit(1)
|
|
71
|
+
|
|
72
|
+
if len(args) > 1:
|
|
73
|
+
data = ' '.join(args[1:])
|
|
74
|
+
else:
|
|
75
|
+
data = sys.stdin.read().strip()
|
|
76
|
+
|
|
77
|
+
if not data:
|
|
78
|
+
print("Error: no input", file=sys.stderr)
|
|
79
|
+
sys.exit(1)
|
|
80
|
+
|
|
81
|
+
try:
|
|
82
|
+
if mode == 'encode':
|
|
83
|
+
result = b58encode(data.encode() if isinstance(data, str) else data)
|
|
84
|
+
else:
|
|
85
|
+
result = b58decode(data)
|
|
86
|
+
try:
|
|
87
|
+
result = result.decode('utf-8')
|
|
88
|
+
except UnicodeDecodeError:
|
|
89
|
+
result = result.hex()
|
|
90
|
+
print(result)
|
|
91
|
+
except Exception as e:
|
|
92
|
+
print(f"Error: {e}", file=sys.stderr)
|
|
93
|
+
sys.exit(1)
|
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""file_find — Find files by name pattern with recursive search.
|
|
3
|
+
|
|
4
|
+
Usage: file_find *.txt # Find all .txt files
|
|
5
|
+
file_find config* --dir . # In current dir
|
|
6
|
+
file_find test_*.py --max 10 # Limit results
|
|
7
|
+
file_find Dockerfile # Exact name match
|
|
8
|
+
|
|
9
|
+
Lightweight alternative to 'find' with simpler syntax.
|
|
10
|
+
Uses fnmatch pattern matching. Zero dependencies.
|
|
11
|
+
"""
|
|
12
|
+
|
|
13
|
+
import os
|
|
14
|
+
import sys
|
|
15
|
+
import fnmatch
|
|
16
|
+
|
|
17
|
+
TOOL_META = {
|
|
18
|
+
"name": "file_find",
|
|
19
|
+
"func": "main",
|
|
20
|
+
"desc": "Find files by name pattern with recursive search",
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
|
|
24
|
+
def main():
|
|
25
|
+
args = sys.argv[1:]
|
|
26
|
+
if not args or args[0] in ('-h', '--help'):
|
|
27
|
+
print(__doc__)
|
|
28
|
+
return
|
|
29
|
+
|
|
30
|
+
pattern = None
|
|
31
|
+
search_dir = os.getcwd()
|
|
32
|
+
max_results = 0
|
|
33
|
+
case_insensitive = False
|
|
34
|
+
type_filter = None # 'f' or 'd'
|
|
35
|
+
|
|
36
|
+
i = 0
|
|
37
|
+
while i < len(args):
|
|
38
|
+
if args[i] == '--dir' and i + 1 < len(args):
|
|
39
|
+
search_dir = os.path.abspath(args[i + 1])
|
|
40
|
+
i += 2
|
|
41
|
+
elif args[i] == '--max' and i + 1 < len(args):
|
|
42
|
+
max_results = int(args[i + 1])
|
|
43
|
+
i += 2
|
|
44
|
+
elif args[i] == '--type' and i + 1 < len(args):
|
|
45
|
+
type_filter = args[i + 1]
|
|
46
|
+
i += 2
|
|
47
|
+
elif args[i] == '--ignore-case' or args[i] == '-i':
|
|
48
|
+
case_insensitive = True
|
|
49
|
+
i += 1
|
|
50
|
+
elif not args[i].startswith('--'):
|
|
51
|
+
pattern = args[i]
|
|
52
|
+
i += 1
|
|
53
|
+
else:
|
|
54
|
+
i += 1
|
|
55
|
+
|
|
56
|
+
if not pattern:
|
|
57
|
+
print("Error: no pattern specified", file=sys.stderr)
|
|
58
|
+
sys.exit(1)
|
|
59
|
+
|
|
60
|
+
if not os.path.isdir(search_dir):
|
|
61
|
+
print(f"Error: directory not found: {search_dir}", file=sys.stderr)
|
|
62
|
+
sys.exit(1)
|
|
63
|
+
|
|
64
|
+
count = 0
|
|
65
|
+
for root, dirs, files in os.walk(search_dir):
|
|
66
|
+
# Skip hidden dirs by default
|
|
67
|
+
dirs[:] = [d for d in dirs if not d.startswith('.')]
|
|
68
|
+
|
|
69
|
+
targets = []
|
|
70
|
+
if type_filter in (None, 'f'):
|
|
71
|
+
targets.extend(files)
|
|
72
|
+
if type_filter in (None, 'd'):
|
|
73
|
+
targets.extend(dirs)
|
|
74
|
+
|
|
75
|
+
for name in targets:
|
|
76
|
+
matched = False
|
|
77
|
+
if case_insensitive:
|
|
78
|
+
matched = fnmatch.fnmatch(name.lower(), pattern.lower())
|
|
79
|
+
else:
|
|
80
|
+
matched = fnmatch.fnmatch(name, pattern)
|
|
81
|
+
|
|
82
|
+
if matched:
|
|
83
|
+
path = os.path.join(root, name)
|
|
84
|
+
print(path)
|
|
85
|
+
count += 1
|
|
86
|
+
if max_results and count >= max_results:
|
|
87
|
+
print(f"--- {count} results (max: {max_results}) ---")
|
|
88
|
+
return
|
|
89
|
+
|
|
90
|
+
if count == 0:
|
|
91
|
+
print(f"No files matching '{pattern}' in {search_dir}")
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""gzip-cli — Gzip/gunzip files from CLI.
|
|
3
|
+
|
|
4
|
+
Usage: gzip-cli <file> # Compress (creates .gz)
|
|
5
|
+
gzip-cli -d <file.gz> # Decompress
|
|
6
|
+
gzip-cli -l <file.gz> # List contents of archive
|
|
7
|
+
cat <file> | gzip-cli # Compress stdin to stdout
|
|
8
|
+
|
|
9
|
+
Zero-dependency (stdlib only — uses gzip).
|
|
10
|
+
"""
|
|
11
|
+
import sys
|
|
12
|
+
import os
|
|
13
|
+
import gzip
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
def main():
|
|
17
|
+
args = sys.argv[1:]
|
|
18
|
+
|
|
19
|
+
if '-h' in args or '--help' in args:
|
|
20
|
+
print(__doc__)
|
|
21
|
+
return
|
|
22
|
+
|
|
23
|
+
decompress = '-d' in args
|
|
24
|
+
list_mode = '-l' in args
|
|
25
|
+
args = [a for a in args if not a.startswith('-') or a in (args and args[0] and args[0])]
|
|
26
|
+
|
|
27
|
+
# Actually re-parse properly
|
|
28
|
+
args = sys.argv[1:]
|
|
29
|
+
decompress = '-d' in args or '--decompress' in args
|
|
30
|
+
list_mode = '-l' in args or '--list' in args
|
|
31
|
+
args = [a for a in args if not a.startswith('-')]
|
|
32
|
+
|
|
33
|
+
# Stdin mode
|
|
34
|
+
if not args:
|
|
35
|
+
if decompress:
|
|
36
|
+
sys.stdout.buffer.write(gzip.decompress(sys.stdin.buffer.read()))
|
|
37
|
+
else:
|
|
38
|
+
sys.stdout.buffer.write(gzip.compress(sys.stdin.buffer.read()))
|
|
39
|
+
return
|
|
40
|
+
|
|
41
|
+
for filepath in args:
|
|
42
|
+
if not os.path.exists(filepath):
|
|
43
|
+
print(f'Error: File not found: {filepath}', file=sys.stderr)
|
|
44
|
+
continue
|
|
45
|
+
|
|
46
|
+
if list_mode:
|
|
47
|
+
with gzip.open(filepath, 'rb') as f:
|
|
48
|
+
content = f.read()
|
|
49
|
+
orig_name = os.path.basename(filepath)
|
|
50
|
+
if orig_name.endswith('.gz'):
|
|
51
|
+
orig_name = orig_name[:-3]
|
|
52
|
+
compressed_size = os.path.getsize(filepath)
|
|
53
|
+
print(f'{orig_name:20s} {len(content):>8} bytes → {compressed_size:>8} bytes (ratio: {compressed_size/len(content)*100:.1f}%)')
|
|
54
|
+
continue
|
|
55
|
+
|
|
56
|
+
if decompress:
|
|
57
|
+
if not filepath.endswith('.gz'):
|
|
58
|
+
print(f'Warning: {filepath} does not end in .gz, decompressing anyway', file=sys.stderr)
|
|
59
|
+
outpath = filepath[:-3] if filepath.endswith('.gz') else filepath + '.out'
|
|
60
|
+
with gzip.open(filepath, 'rb') as f_in:
|
|
61
|
+
content = f_in.read()
|
|
62
|
+
with open(outpath, 'wb') as f_out:
|
|
63
|
+
f_out.write(content)
|
|
64
|
+
print(f'{filepath} → {outpath} ({len(content)} bytes)')
|
|
65
|
+
else:
|
|
66
|
+
outpath = filepath + '.gz'
|
|
67
|
+
with open(filepath, 'rb') as f_in:
|
|
68
|
+
content = f_in.read()
|
|
69
|
+
with gzip.open(outpath, 'wb') as f_out:
|
|
70
|
+
f_out.write(content)
|
|
71
|
+
print(f'{filepath} → {outpath} ({os.path.getsize(outpath)} bytes, ratio: {os.path.getsize(outpath)/len(content)*100:.1f}%)')
|
|
72
|
+
|
|
73
|
+
|
|
74
|
+
TOOL_META = {
|
|
75
|
+
"name": "gzip-cli",
|
|
76
|
+
"func": "main",
|
|
77
|
+
"desc": "Gzip/gunzip files — compress, decompress, list archive contents",
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
if __name__ == '__main__':
|
|
81
|
+
main()
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""ip_info — Show public IP address and geolocation info.
|
|
3
|
+
|
|
4
|
+
Usage: ip_info # Your public IP
|
|
5
|
+
ip_info 8.8.8.8 # Lookup specific IP
|
|
6
|
+
ip_info --json # JSON output
|
|
7
|
+
|
|
8
|
+
Uses ip-api.com for geolocation data.
|
|
9
|
+
"""
|
|
10
|
+
|
|
11
|
+
import sys
|
|
12
|
+
import json
|
|
13
|
+
import urllib.request
|
|
14
|
+
import urllib.error
|
|
15
|
+
|
|
16
|
+
TOOL_META = {
|
|
17
|
+
"name": "ip_info",
|
|
18
|
+
"func": "main",
|
|
19
|
+
"desc": "Show public IP address and geolocation info",
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
|
|
23
|
+
def main():
|
|
24
|
+
args = sys.argv[1:]
|
|
25
|
+
ip = None
|
|
26
|
+
json_mode = False
|
|
27
|
+
|
|
28
|
+
for arg in args:
|
|
29
|
+
if arg in ('-h', '--help'):
|
|
30
|
+
print(__doc__)
|
|
31
|
+
return
|
|
32
|
+
elif arg == '--json' or arg == '-j':
|
|
33
|
+
json_mode = True
|
|
34
|
+
elif not arg.startswith('--'):
|
|
35
|
+
ip = arg
|
|
36
|
+
else:
|
|
37
|
+
print(f"Unknown flag: {arg}", file=sys.stderr)
|
|
38
|
+
sys.exit(1)
|
|
39
|
+
|
|
40
|
+
target = ip or ''
|
|
41
|
+
url = f"http://ip-api.com/json/{target}?fields=status,message,query,country,regionName,city,zip,lat,lon,isp,org,as,timezone"
|
|
42
|
+
|
|
43
|
+
try:
|
|
44
|
+
resp = urllib.request.urlopen(url, timeout=10)
|
|
45
|
+
data = json.loads(resp.read().decode())
|
|
46
|
+
except urllib.error.URLError as e:
|
|
47
|
+
print(f"Error: {e.reason}", file=sys.stderr)
|
|
48
|
+
sys.exit(1)
|
|
49
|
+
except json.JSONDecodeError:
|
|
50
|
+
print("Error: invalid response from API", file=sys.stderr)
|
|
51
|
+
sys.exit(1)
|
|
52
|
+
|
|
53
|
+
if data.get('status') == 'fail':
|
|
54
|
+
print(f"Error: {data.get('message', 'unknown')}", file=sys.stderr)
|
|
55
|
+
sys.exit(1)
|
|
56
|
+
|
|
57
|
+
if json_mode:
|
|
58
|
+
print(json.dumps(data, indent=2))
|
|
59
|
+
return
|
|
60
|
+
|
|
61
|
+
print(f" IP: {data['query']}")
|
|
62
|
+
print(f" Country: {data.get('country', 'N/A')}")
|
|
63
|
+
print(f" Region: {data.get('regionName', 'N/A')}")
|
|
64
|
+
print(f" City: {data.get('city', 'N/A')}")
|
|
65
|
+
print(f" ZIP: {data.get('zip', 'N/A')}")
|
|
66
|
+
print(f" Lat/Lon: {data.get('lat', '')}, {data.get('lon', '')}")
|
|
67
|
+
print(f" ISP: {data.get('isp', 'N/A')}")
|
|
68
|
+
print(f" Org: {data.get('org', 'N/A')}")
|
|
69
|
+
print(f" AS: {data.get('as', 'N/A')}")
|
|
70
|
+
print(f" Timezone: {data.get('timezone', 'N/A')}")
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""json-diff — Deep diff two JSON files, show paths that differ.
|
|
3
|
+
|
|
4
|
+
Usage: json-diff <file1> <file2>
|
|
5
|
+
json-diff -- <json1> <json2>
|
|
6
|
+
|
|
7
|
+
Zero-dependency (stdlib only).
|
|
8
|
+
"""
|
|
9
|
+
import sys
|
|
10
|
+
import json
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
def deep_diff(a, b, path=''):
|
|
14
|
+
"""Recursively compare two values and yield differing paths."""
|
|
15
|
+
if type(a) != type(b):
|
|
16
|
+
yield f'{path}: type mismatch ({type(a).__name__} vs {type(b).__name__})'
|
|
17
|
+
yield f' - {json.dumps(a, ensure_ascii=False)[:200]}'
|
|
18
|
+
yield f' + {json.dumps(b, ensure_ascii=False)[:200]}'
|
|
19
|
+
return
|
|
20
|
+
|
|
21
|
+
if isinstance(a, dict):
|
|
22
|
+
all_keys = set(a) | set(b)
|
|
23
|
+
for k in sorted(all_keys):
|
|
24
|
+
new_path = f'{path}.{k}' if path else k
|
|
25
|
+
if k not in a:
|
|
26
|
+
yield f'{new_path}: added'
|
|
27
|
+
yield f' + {json.dumps(b[k], ensure_ascii=False)[:200]}'
|
|
28
|
+
elif k not in b:
|
|
29
|
+
yield f'{new_path}: removed'
|
|
30
|
+
yield f' - {json.dumps(a[k], ensure_ascii=False)[:200]}'
|
|
31
|
+
else:
|
|
32
|
+
yield from deep_diff(a[k], b[k], new_path)
|
|
33
|
+
elif isinstance(a, list):
|
|
34
|
+
max_len = max(len(a), len(b))
|
|
35
|
+
for i in range(max_len):
|
|
36
|
+
new_path = f'{path}[{i}]'
|
|
37
|
+
if i >= len(a):
|
|
38
|
+
yield f'{new_path}: added'
|
|
39
|
+
yield f' + {json.dumps(b[i], ensure_ascii=False)[:200]}'
|
|
40
|
+
elif i >= len(b):
|
|
41
|
+
yield f'{new_path}: removed'
|
|
42
|
+
yield f' - {json.dumps(a[i], ensure_ascii=False)[:200]}'
|
|
43
|
+
else:
|
|
44
|
+
yield from deep_diff(a[i], b[i], new_path)
|
|
45
|
+
else:
|
|
46
|
+
if a != b:
|
|
47
|
+
yield f'{path}: changed'
|
|
48
|
+
yield f' - {json.dumps(a, ensure_ascii=False)[:200]}'
|
|
49
|
+
yield f' + {json.dumps(b, ensure_ascii=False)[:200]}'
|
|
50
|
+
|
|
51
|
+
|
|
52
|
+
def main():
|
|
53
|
+
args = sys.argv[1:]
|
|
54
|
+
|
|
55
|
+
if '-h' in args or '--help' in args or len(args) != 2:
|
|
56
|
+
print(__doc__)
|
|
57
|
+
return
|
|
58
|
+
|
|
59
|
+
try:
|
|
60
|
+
if args[0] == '--':
|
|
61
|
+
data1 = json.loads(args[1])
|
|
62
|
+
else:
|
|
63
|
+
with open(args[0]) as f:
|
|
64
|
+
data1 = json.load(f)
|
|
65
|
+
except Exception as e:
|
|
66
|
+
print(f'Error reading {args[0]}: {e}', file=sys.stderr)
|
|
67
|
+
sys.exit(1)
|
|
68
|
+
|
|
69
|
+
try:
|
|
70
|
+
if args[1] == '--':
|
|
71
|
+
data2 = json.loads(args[0])
|
|
72
|
+
else:
|
|
73
|
+
with open(args[1]) as f:
|
|
74
|
+
data2 = json.load(f)
|
|
75
|
+
except Exception as e:
|
|
76
|
+
print(f'Error reading {args[1]}: {e}', file=sys.stderr)
|
|
77
|
+
sys.exit(1)
|
|
78
|
+
|
|
79
|
+
diffs = list(deep_diff(data1, data2))
|
|
80
|
+
if diffs:
|
|
81
|
+
print(f'{len(diffs) // 2} difference(s) found:')
|
|
82
|
+
print()
|
|
83
|
+
for line in diffs:
|
|
84
|
+
print(line)
|
|
85
|
+
else:
|
|
86
|
+
print('No differences — files are identical.')
|
|
87
|
+
|
|
88
|
+
|
|
89
|
+
TOOL_META = {
|
|
90
|
+
"name": "json-diff",
|
|
91
|
+
"func": "main",
|
|
92
|
+
"desc": "Deep diff two JSON files, show paths that differ",
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
if __name__ == '__main__':
|
|
96
|
+
main()
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""json_merge — Merge multiple JSON files into one.
|
|
3
|
+
|
|
4
|
+
Usage: json_merge file1.json file2.json
|
|
5
|
+
json_merge *.json --output merged.json
|
|
6
|
+
json_merge base.json override.json --strategy deep
|
|
7
|
+
|
|
8
|
+
Merge strategies:
|
|
9
|
+
- shallow (default): top-level keys only. Later files override earlier.
|
|
10
|
+
- deep: recursive merge of nested dicts.
|
|
11
|
+
- array: concatenate arrays.
|
|
12
|
+
"""
|
|
13
|
+
|
|
14
|
+
import json
|
|
15
|
+
import sys
|
|
16
|
+
import os
|
|
17
|
+
from copy import deepcopy
|
|
18
|
+
|
|
19
|
+
TOOL_META = {
|
|
20
|
+
"name": "json_merge",
|
|
21
|
+
"func": "main",
|
|
22
|
+
"desc": "Merge multiple JSON files into one",
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
def deep_merge(a, b):
|
|
27
|
+
"""Recursively merge b into a."""
|
|
28
|
+
result = deepcopy(a)
|
|
29
|
+
for key in b:
|
|
30
|
+
if key in result and isinstance(result[key], dict) and isinstance(b[key], dict):
|
|
31
|
+
result[key] = deep_merge(result[key], b[key])
|
|
32
|
+
else:
|
|
33
|
+
result[key] = deepcopy(b[key])
|
|
34
|
+
return result
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
def array_merge(a, b):
|
|
38
|
+
"""Merge arrays by concatenation."""
|
|
39
|
+
if isinstance(a, dict) and isinstance(b, dict):
|
|
40
|
+
result = deepcopy(a)
|
|
41
|
+
for key in b:
|
|
42
|
+
if key in result and isinstance(result[key], list) and isinstance(b[key], list):
|
|
43
|
+
result[key] = result[key] + b[key]
|
|
44
|
+
elif key in result and isinstance(result[key], dict) and isinstance(b[key], dict):
|
|
45
|
+
result[key] = deep_merge(result[key], b[key])
|
|
46
|
+
else:
|
|
47
|
+
result[key] = deepcopy(b[key])
|
|
48
|
+
return result
|
|
49
|
+
elif isinstance(a, list) and isinstance(b, list):
|
|
50
|
+
return a + b
|
|
51
|
+
return b
|
|
52
|
+
|
|
53
|
+
|
|
54
|
+
def main():
|
|
55
|
+
args = sys.argv[1:]
|
|
56
|
+
files = []
|
|
57
|
+
strategy = 'shallow'
|
|
58
|
+
output = None
|
|
59
|
+
|
|
60
|
+
i = 0
|
|
61
|
+
while i < len(args):
|
|
62
|
+
if args[i] in ('-h', '--help'):
|
|
63
|
+
print(__doc__)
|
|
64
|
+
return
|
|
65
|
+
elif args[i] == '--strategy' and i + 1 < len(args):
|
|
66
|
+
strategy = args[i + 1].lower()
|
|
67
|
+
i += 2
|
|
68
|
+
elif args[i] == '--output' and i + 1 < len(args):
|
|
69
|
+
output = args[i + 1]
|
|
70
|
+
i += 2
|
|
71
|
+
elif not args[i].startswith('--'):
|
|
72
|
+
files.append(args[i])
|
|
73
|
+
i += 1
|
|
74
|
+
else:
|
|
75
|
+
i += 1
|
|
76
|
+
|
|
77
|
+
if len(files) < 2:
|
|
78
|
+
print("Error: need at least 2 JSON files", file=sys.stderr)
|
|
79
|
+
sys.exit(1)
|
|
80
|
+
|
|
81
|
+
data = []
|
|
82
|
+
for path in files:
|
|
83
|
+
if not os.path.exists(path):
|
|
84
|
+
print(f"Error: file not found: {path}", file=sys.stderr)
|
|
85
|
+
sys.exit(1)
|
|
86
|
+
with open(path) as f:
|
|
87
|
+
data.append(json.load(f))
|
|
88
|
+
|
|
89
|
+
result = data[0]
|
|
90
|
+
for d in data[1:]:
|
|
91
|
+
if strategy == 'deep':
|
|
92
|
+
result = deep_merge(result, d)
|
|
93
|
+
elif strategy == 'array':
|
|
94
|
+
result = array_merge(result, d)
|
|
95
|
+
else:
|
|
96
|
+
# shallow
|
|
97
|
+
result.update(d)
|
|
98
|
+
|
|
99
|
+
if output:
|
|
100
|
+
with open(output, 'w') as f:
|
|
101
|
+
json.dump(result, f, indent=2)
|
|
102
|
+
print(f"Merged {len(files)} files -> {output}")
|
|
103
|
+
else:
|
|
104
|
+
print(json.dumps(result, indent=2))
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""jwt-decode — Decode JWT tokens (header + payload, no verification).
|
|
3
|
+
|
|
4
|
+
Usage: jwt-decode <token>
|
|
5
|
+
echo <token> | jwt-decode
|
|
6
|
+
|
|
7
|
+
Zero-dependency (stdlib only — uses base64 + json).
|
|
8
|
+
"""
|
|
9
|
+
import sys
|
|
10
|
+
import base64
|
|
11
|
+
import json
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
def decode_part(data: str) -> str:
|
|
15
|
+
# Add padding
|
|
16
|
+
padding = 4 - len(data) % 4
|
|
17
|
+
if padding != 4:
|
|
18
|
+
data += '=' * padding
|
|
19
|
+
try:
|
|
20
|
+
decoded = base64.urlsafe_b64decode(data)
|
|
21
|
+
return decoded.decode('utf-8')
|
|
22
|
+
except Exception:
|
|
23
|
+
return f'[binary data: {len(data)} bytes]'
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
def main():
|
|
27
|
+
args = sys.argv[1:]
|
|
28
|
+
|
|
29
|
+
if '-h' in args or '--help' in args:
|
|
30
|
+
print(__doc__)
|
|
31
|
+
return
|
|
32
|
+
|
|
33
|
+
if args:
|
|
34
|
+
token = args[0]
|
|
35
|
+
else:
|
|
36
|
+
token = sys.stdin.read().strip()
|
|
37
|
+
|
|
38
|
+
if not token:
|
|
39
|
+
print('Error: No JWT token provided', file=sys.stderr)
|
|
40
|
+
sys.exit(1)
|
|
41
|
+
|
|
42
|
+
parts = token.split('.')
|
|
43
|
+
if len(parts) != 3:
|
|
44
|
+
print('Error: Invalid JWT — expected 3 dot-separated parts', file=sys.stderr)
|
|
45
|
+
sys.exit(1)
|
|
46
|
+
|
|
47
|
+
print('=== JWT Decode (no verification) ===')
|
|
48
|
+
print()
|
|
49
|
+
|
|
50
|
+
# Header
|
|
51
|
+
header_raw = decode_part(parts[0])
|
|
52
|
+
try:
|
|
53
|
+
header = json.loads(header_raw)
|
|
54
|
+
print(f'Header: {json.dumps(header, indent=2)}')
|
|
55
|
+
except json.JSONDecodeError:
|
|
56
|
+
print(f'Header (raw): {header_raw}')
|
|
57
|
+
|
|
58
|
+
print()
|
|
59
|
+
|
|
60
|
+
# Payload
|
|
61
|
+
payload_raw = decode_part(parts[1])
|
|
62
|
+
try:
|
|
63
|
+
payload = json.loads(payload_raw)
|
|
64
|
+
print(f'Payload: {json.dumps(payload, indent=2)}')
|
|
65
|
+
except json.JSONDecodeError:
|
|
66
|
+
print(f'Payload (raw): {payload_raw}')
|
|
67
|
+
|
|
68
|
+
print()
|
|
69
|
+
print(f'Signature: {parts[2][:40]}... ({len(parts[2])} chars)')
|
|
70
|
+
print()
|
|
71
|
+
print('WARNING: Decoded only — signature NOT verified.')
|
|
72
|
+
|
|
73
|
+
|
|
74
|
+
TOOL_META = {
|
|
75
|
+
"name": "jwt-decode",
|
|
76
|
+
"func": "main",
|
|
77
|
+
"desc": "Decode JWT tokens (header + payload, no verification)",
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
if __name__ == '__main__':
|
|
81
|
+
main()
|