| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268 |
- #!/usr/bin/env python
- # Mode: -*- python -*-
- #
- # Copyright (c) 2015-2017, 2019-2024 by Rocky Bernstein
- # Copyright (c) 2000-2002 by hartmut Goebel <h.goebel@crazy-compilers.com>
- #
- import os
- import sys
- from typing import List
- import click
- from xdis.version_info import version_tuple_to_str
- from decompyle3.main import main, status_msg
- from decompyle3.version import __version__
- case_sensitive = {"case_sensitive": False}
- program = "decompyle3"
- def usage():
- print(__doc__)
- sys.exit(1)
- @click.command()
- @click.option(
- "--asm++/--no-asm++",
- "-A",
- "asm_plus",
- default=False,
- help="show xdis assembler and tokenized assembler",
- )
- @click.option("--asm/--no-asm", "-a", default=False)
- @click.option("--grammar/--no-grammar", "-g", "show_grammar", default=False)
- @click.option("--tree/--no-tree", "-t", default=False)
- @click.option(
- "--tree++/--no-tree++",
- "-T",
- "tree_plus",
- default=False,
- help="show parse tree and Abstract Syntax Tree",
- )
- @click.option(
- "--linemaps/--no-linemaps",
- default=False,
- help="show line number correspondencies between byte-code "
- "and generated source output",
- )
- @click.option(
- "--verify",
- type=click.Choice(["run", "syntax"]),
- default=None,
- )
- @click.option(
- "--recurse/--no-recurse",
- "-r",
- "recurse_dirs",
- default=False,
- )
- @click.option(
- "--output",
- "-o",
- "outfile",
- type=click.Path(
- exists=True, file_okay=True, dir_okay=True, writable=True, resolve_path=True
- ),
- required=False,
- )
- @click.version_option(version=__version__)
- @click.option(
- "--start-offset",
- "start_offset",
- default=0,
- help="start decomplation at offset; default is 0 or the starting offset.",
- )
- @click.version_option(version=__version__)
- @click.option(
- "--stop-offset",
- "stop_offset",
- default=-1,
- help="stop decomplation when seeing an offset greater or equal to this; default is "
- "-1 which indicates no stopping point.",
- )
- @click.argument("files", nargs=-1, type=click.Path(readable=True), required=True)
- def main_bin(
- asm_plus: bool,
- asm: bool,
- show_grammar,
- tree: bool,
- tree_plus: bool,
- linemaps: bool,
- verify,
- recurse_dirs: bool,
- outfile,
- start_offset: int,
- stop_offset: int,
- files: List[str],
- ):
- """
- Cross Python bytecode decompiler for Python 3.7-3.8 bytecode
- """
- version_tuple = sys.version_info[0:2]
- if version_tuple < (3, 7):
- print(
- f"Error: {program} runs from Python 3.7 or greater."
- f""" \n\tYou have version: {version_tuple_to_str()}."""
- )
- sys.exit(-1)
- out_base = None
- source_paths: List[str] = []
- # timestamp = False
- # timestampfmt = "# %Y.%m.%d %H:%M:%S %Z"
- pyc_paths = files
- # Expand directory if "recurse" was specified.
- if recurse_dirs:
- expanded_files = []
- for f in pyc_paths:
- if os.path.isdir(f):
- for root, _, dir_files in os.walk(f):
- for df in dir_files:
- if df.endswith(".pyc") or df.endswith(".pyo"):
- expanded_files.append(os.path.join(root, df))
- pyc_paths = expanded_files
- # argl, commonprefix works on strings, not on path parts,
- # thus we must handle the case with files in 'some/classes'
- # and 'some/cmds'
- src_base = os.path.commonprefix(pyc_paths)
- if src_base[-1:] != os.sep:
- src_base = os.path.dirname(src_base)
- if src_base:
- sb_len = len(os.path.join(src_base, ""))
- pyc_paths = [f[sb_len:] for f in pyc_paths]
- if not pyc_paths and not source_paths:
- print("No input files given to decompile", file=sys.stderr)
- usage()
- if outfile == "-":
- outfile = None # use stdout
- elif outfile and os.path.isdir(outfile):
- out_base = outfile
- outfile = None
- elif outfile and len(pyc_paths) > 1:
- out_base = outfile
- outfile = None
- # A second -a turns show_asm="after" into show_asm="before"
- if asm_plus or asm:
- asm_opt = "both" if asm_plus else "after"
- else:
- asm_opt = None
- # if timestamp:
- # print(time.strftime(timestampfmt))
- show_grammar = {
- "rules": False,
- "transition": False,
- "reduce": show_grammar,
- "errorstack": "full",
- "context": True,
- "dups": False,
- }
- numproc = 1
- if numproc <= 1:
- show_ast = {"before": tree or tree_plus, "after": tree_plus}
- try:
- result = main(
- src_base,
- out_base,
- pyc_paths,
- source_paths,
- outfile,
- showasm=asm_opt,
- showgrammar=show_grammar,
- showast=show_ast,
- do_verify=verify,
- do_linemaps=linemaps,
- start_offset=start_offset,
- stop_offset=stop_offset,
- )
- if len(pyc_paths) > 1:
- mess = status_msg(verify, *result)
- print("# " + mess)
- pass
- except ImportError as e:
- print(str(e))
- sys.exit(2)
- except KeyboardInterrupt:
- pass
- else:
- from multiprocessing import Process, Queue
- from queue import Empty
- fqueue = Queue(len(pyc_paths) + numproc)
- for f in pyc_paths:
- fqueue.put(f)
- for i in range(numproc):
- fqueue.put(None)
- rqueue = Queue(numproc)
- def process_func():
- (tot_files, okay_files, failed_files, verify_failed_files) = (
- 0,
- 0,
- 0,
- 0,
- )
- try:
- while True:
- f = fqueue.get()
- if f is None:
- break
- (t, o, f, v) = main(src_base, out_base, [f], [], outfile)
- tot_files += t
- okay_files += o
- failed_files += f
- verify_failed_files += v
- except (Empty, KeyboardInterrupt):
- pass
- rqueue.put((tot_files, okay_files, failed_files, verify_failed_files))
- rqueue.close()
- try:
- procs = [Process(target=process_func) for _ in range(numproc)]
- for p in procs:
- p.start()
- for p in procs:
- p.join()
- (tot_files, okay_files, failed_files, verify_failed_files) = (
- 0,
- 0,
- 0,
- 0,
- )
- try:
- while True:
- (t, o, f, v) = rqueue.get(False)
- tot_files += t
- okay_files += o
- failed_files += f
- verify_failed_files += v
- except Empty:
- pass
- print(
- "# decompiled %i files: %i okay, %i failed, %i verify failed"
- % (tot_files, okay_files, failed_files, verify_failed_files)
- )
- except (KeyboardInterrupt, OSError):
- pass
- # if timestamp:
- # print(time.strftime(timestampfmt))
- return
- if __name__ == "__main__":
- main_bin()
|