You've already forked python-uncompyle6
mirror of
https://github.com/rocky/python-uncompyle6.git
synced 2025-08-04 01:09:52 +08:00
Compare commits
375 Commits
releaes-3.
...
3.5.2
Author | SHA1 | Date | |
---|---|---|---|
|
3e3dd87c3b | ||
|
edbbefb57d | ||
|
6546bbdaf9 | ||
|
825ed3fef9 | ||
|
7d9c4ce8ca | ||
|
fdac1e3c46 | ||
|
daab1e8610 | ||
|
b8f4dca505 | ||
|
99b8a99ffa | ||
|
8c879c02de | ||
|
d11a9ea126 | ||
|
4926474efc | ||
|
eba5226a04 | ||
|
8d0ff367d8 | ||
|
c6ddefcef5 | ||
|
301464d646 | ||
|
d5b52d44e0 | ||
|
322f491c83 | ||
|
2987d6a72b | ||
|
7609165967 | ||
|
655162a05e | ||
|
ca7f483dbb | ||
|
e713169bdf | ||
|
cc856e2b95 | ||
|
d696443eb2 | ||
|
a5e7eb19c6 | ||
|
6659fffc0d | ||
|
1868b566a1 | ||
|
791274c45d | ||
|
4327ee98e6 | ||
|
2717a5e302 | ||
|
77dd3b8d50 | ||
|
2cfc60fbd7 | ||
|
daa424cf0c | ||
|
7b68c9c838 | ||
|
df5df9364c | ||
|
f1496cad4d | ||
|
a3a15414d3 | ||
|
9874553fb4 | ||
|
d21d93fd84 | ||
|
dbf2729f76 | ||
|
047e27c966 | ||
|
6a81a752a7 | ||
|
44f0ba0efb | ||
|
bc8907e752 | ||
|
4e9d8783d1 | ||
|
47c847644e | ||
|
af2ed31871 | ||
|
49de5b5c9d | ||
|
c8252ca9cb | ||
|
0441fbc616 | ||
|
1b4335edf1 | ||
|
2e36551c02 | ||
|
fff6f82dd7 | ||
|
7c8f3cc9ec | ||
|
78a595c8cf | ||
|
cda0154594 | ||
|
d852f23962 | ||
|
065fd13b81 | ||
|
659f37585b | ||
|
bc18fcf7fa | ||
|
144f52da8e | ||
|
9f250b49ee | ||
|
4abdffecb9 | ||
|
1419acf019 | ||
|
bdc24d7f51 | ||
|
07ec8fa1fb | ||
|
04c2240d63 | ||
|
96dcdfd744 | ||
|
82ea77c592 | ||
|
4f0e580438 | ||
|
09cc0d775a | ||
|
2da5a5649f | ||
|
373916f57c | ||
|
1c0c54991e | ||
|
5de5d2357f | ||
|
6d296f11c9 | ||
|
0ae4acff9e | ||
|
40c2f2962c | ||
|
dae195e36e | ||
|
2c503d5a14 | ||
|
eed4c1025b | ||
|
86c1d12e69 | ||
|
61a367b0ae | ||
|
dba6d24361 | ||
|
0b111f1568 | ||
|
c666e2dc3d | ||
|
0a5fcc51d8 | ||
|
ade9f7a182 | ||
|
d41ef3e5dc | ||
|
ebb0342b38 | ||
|
f17ebf42a9 | ||
|
85dcce4ff2 | ||
|
bc9a16eaac | ||
|
d08d183fc8 | ||
|
0b3d6b8add | ||
|
5cb46c2ed3 | ||
|
163e47fb49 | ||
|
0cf32f1b70 | ||
|
f0f9676f52 | ||
|
be610aa6b3 | ||
|
1494bb2049 | ||
|
d62dc3daac | ||
|
5ad51707e3 | ||
|
f28c255804 | ||
|
315965300f | ||
|
9bd85fe5a0 | ||
|
c6e3168c31 | ||
|
7969b19c2b | ||
|
f8bfde4a8e | ||
|
e2b309fa30 | ||
|
1ebfde6927 | ||
|
73619de3f5 | ||
|
600cd0b8ee | ||
|
90a2ed2c9e | ||
|
8728cb6a99 | ||
|
8daedaf063 | ||
|
4a4a20995e | ||
|
c923ce9afe | ||
|
88901c6901 | ||
|
31f7d14eab | ||
|
388d1da970 | ||
|
66294d54f7 | ||
|
55783c2712 | ||
|
fd580f3c60 | ||
|
a781006ff1 | ||
|
0be3d5a530 | ||
|
bc3cd0102b | ||
|
6e6d590268 | ||
|
71bdc8dc6a | ||
|
71735ca7ef | ||
|
9f121ef00c | ||
|
2e01f42f65 | ||
|
3f9a862277 | ||
|
cc531cf90a | ||
|
c7124ad9ca | ||
|
44a4aab0a7 | ||
|
2a52982d52 | ||
|
56e5e8dcef | ||
|
ecab7d7b09 | ||
|
e39a902e56 | ||
|
e2914ed552 | ||
|
f425db33b7 | ||
|
68c5b2338f | ||
|
e55a0410c9 | ||
|
0fe8961418 | ||
|
8cd331a32b | ||
|
4c76931807 | ||
|
7b7f794913 | ||
|
50e46531ce | ||
|
67ef34977f | ||
|
32c7b8f23d | ||
|
2f06d1eeb0 | ||
|
999f1fb0f9 | ||
|
76eef9a149 | ||
|
c8b945fb56 | ||
|
a1e7c16dbe | ||
|
35f14e4357 | ||
|
49d1a50354 | ||
|
0dc19a8fdd | ||
|
f6aa8b2baf | ||
|
887a006849 | ||
|
e26c7407a0 | ||
|
69823af553 | ||
|
e96498eaf0 | ||
|
9d6d6a355d | ||
|
04c53c1086 | ||
|
96866f94a7 | ||
|
d371839c99 | ||
|
24afe072b7 | ||
|
e2d7f01298 | ||
|
b39112b601 | ||
|
20b513fc81 | ||
|
d369017122 | ||
|
6675ea2cd0 | ||
|
4b82806d6c | ||
|
3c06b82931 | ||
|
c680416f92 | ||
|
58c8fe5a66 | ||
|
aea1adeb85 | ||
|
c871a4ecc5 | ||
|
cd9eca7bff | ||
|
002720988c | ||
|
08f23567a6 | ||
|
43348d7d24 | ||
|
164e9d4b5c | ||
|
37e4754268 | ||
|
c3257a9b79 | ||
|
70b0704967 | ||
|
76dcaf9bf0 | ||
|
21fd506fbb | ||
|
efe0914814 | ||
|
5981c7eae9 | ||
|
36ef1607af | ||
|
b2d97f9847 | ||
|
24ba5d7f40 | ||
|
eae3f0d77b | ||
|
a54fba7993 | ||
|
719d2d7232 | ||
|
e82cabc278 | ||
|
9ab086b207 | ||
|
4022e80d6d | ||
|
9811c5bc42 | ||
|
354796fffd | ||
|
ab696b316a | ||
|
2f99da8199 | ||
|
fd5f4fa5b8 | ||
|
8e4168674d | ||
|
c8fc6a704c | ||
|
622d6f849c | ||
|
10d8aed4c0 | ||
|
86fd5dbf7a | ||
|
9fe1752359 | ||
|
48ae7a6964 | ||
|
117b4ff4f1 | ||
|
e9002038f8 | ||
|
9d47b99932 | ||
|
59b012df6f | ||
|
44d7cbcf6f | ||
|
9bae73679f | ||
|
ceebe9ab60 | ||
|
b7e22b4530 | ||
|
c7b20edba0 | ||
|
64e35b09db | ||
|
a0d4daf5ff | ||
|
afa6a00db8 | ||
|
d8f0d31475 | ||
|
dd76a6f253 | ||
|
cb40caa73c | ||
|
fd59879510 | ||
|
c9cae2d09e | ||
|
af209dc142 | ||
|
ad419e0ed9 | ||
|
ee5c7da790 | ||
|
39c12704a8 | ||
|
3b3fc09b60 | ||
|
f7697ccd7b | ||
|
e364499bb9 | ||
|
9db59f1b80 | ||
|
a5cdb50154 | ||
|
792ef5b5b8 | ||
|
47ed0795b2 | ||
|
cccf33573b | ||
|
3c3e5c82fc | ||
|
436260dc9a | ||
|
8f0674706b | ||
|
01cc184716 | ||
|
2771cb46ab | ||
|
9ed4326f7e | ||
|
e3b10b62d7 | ||
|
59b8f18486 | ||
|
bcf6939312 | ||
|
3b7f49c01d | ||
|
ae976e991a | ||
|
8fe6309650 | ||
|
4c4aa393df | ||
|
a8b8c2908c | ||
|
cb406e2581 | ||
|
20b16c44ff | ||
|
3abe8d11d3 | ||
|
26140934da | ||
|
b62752eca1 | ||
|
9db446d928 | ||
|
46acb74745 | ||
|
44e1288e2f | ||
|
ce9270dda0 | ||
|
3d732db3cc | ||
|
009a74da7d | ||
|
251eb6da1b | ||
|
8b5e0f49f8 | ||
|
1cc08d9598 | ||
|
d99e78d46d | ||
|
b94cce7b12 | ||
|
fe786b2b95 | ||
|
bf56fbeeec | ||
|
6d8d9fd83b | ||
|
78ca6a0c1f | ||
|
86dd321256 | ||
|
4db364f701 | ||
|
c03b039714 | ||
|
d97509495e | ||
|
4d793ba1b2 | ||
|
590d2f44f1 | ||
|
e875b79a75 | ||
|
b57ca392a2 | ||
|
a132e2ace6 | ||
|
b05500dd49 | ||
|
65307f257c | ||
|
8909fe8d37 | ||
|
733a44e22f | ||
|
f2f17740ee | ||
|
393e5c9303 | ||
|
8c611476fe | ||
|
6df65a87bc | ||
|
bb94c7f5bc | ||
|
8e9ce0be31 | ||
|
bc49469704 | ||
|
5905cce1de | ||
|
af816c9e60 | ||
|
82a3419eb2 | ||
|
46ca21596f | ||
|
6e753b8743 | ||
|
0007abf827 | ||
|
7ecfb74e9a | ||
|
2813e2212f | ||
|
e2c5a79346 | ||
|
293e7b0367 | ||
|
32bc017e2e | ||
|
ce7015f382 | ||
|
4cc53f2307 | ||
|
257bbc892f | ||
|
fac365f216 | ||
|
f54cf20d9d | ||
|
03d23328eb | ||
|
c074107504 | ||
|
a981db884c | ||
|
c5d7944e65 | ||
|
43dbf9b878 | ||
|
efa964f7c9 | ||
|
5c58a4816f | ||
|
132a9acdb4 | ||
|
9186a3fc44 | ||
|
05db6194ec | ||
|
3730946a1a | ||
|
f1b69a8a28 | ||
|
0e5eb954b2 | ||
|
7d9286b353 | ||
|
0de99e5d44 | ||
|
52af2ba32a | ||
|
bd0db6c539 | ||
|
8663b4ca52 | ||
|
b2dd58a85e | ||
|
97cb193a71 | ||
|
e6e60cb49d | ||
|
2ea8c3b1b1 | ||
|
701d2af54e | ||
|
8a4189bc0e | ||
|
0c4ab699b5 | ||
|
8e11c53064 | ||
|
b4c66d4307 | ||
|
53968e535f | ||
|
d2381fbe11 | ||
|
d413ebe0e1 | ||
|
f96522e18e | ||
|
50d50af2ee | ||
|
4dc2897cdc | ||
|
47fb80494d | ||
|
830e19e3e6 | ||
|
c5cfd36a61 | ||
|
9b550b9dda | ||
|
400943bb6a | ||
|
f89ba40147 | ||
|
32c611a315 | ||
|
5d91e96358 | ||
|
44edf1d7db | ||
|
a891aa0706 | ||
|
5e1340a2fc | ||
|
94eff282f8 | ||
|
7f65a8a6dd | ||
|
cfe7feed4d | ||
|
b3a20896b2 | ||
|
1425476018 | ||
|
59c77f103d | ||
|
726045a05e | ||
|
49e354375e | ||
|
f3d86e0708 | ||
|
820283827f | ||
|
8b65cc7275 | ||
|
1e47f47527 | ||
|
19a95be3ef | ||
|
5a6550b353 | ||
|
82fb9426af | ||
|
98b91db8e6 | ||
|
ce3f815b08 | ||
|
c447b9cfa9 |
@@ -1,11 +1,3 @@
|
|||||||
# This configuration was automatically generated from a CircleCI 1.0 config.
|
|
||||||
# It should include any build commands you had along with commands that CircleCI
|
|
||||||
# inferred from your project structure. We strongly recommend you read all the
|
|
||||||
# comments in this file to understand the structure of CircleCI 2.0, as the idiom
|
|
||||||
# for configuration has changed substantially in 2.0 to allow arbitrary jobs rather
|
|
||||||
# than the prescribed lifecycle of 1.0. In general, we recommend using this generated
|
|
||||||
# configuration as a reference rather than using it in production, though in most
|
|
||||||
# cases it should duplicate the execution of your original 1.0 config.
|
|
||||||
version: 2
|
version: 2
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
@@ -27,9 +19,9 @@ jobs:
|
|||||||
# VM instead of a container) see https://circleci.com/docs/2.0/executor-types/
|
# VM instead of a container) see https://circleci.com/docs/2.0/executor-types/
|
||||||
# To see the list of pre-built images that CircleCI provides for most common languages see
|
# To see the list of pre-built images that CircleCI provides for most common languages see
|
||||||
# https://circleci.com/docs/2.0/circleci-images/
|
# https://circleci.com/docs/2.0/circleci-images/
|
||||||
docker:
|
machine:
|
||||||
- image: circleci/build-image:ubuntu-14.04-XXL-upstart-1189-5614f37
|
python:
|
||||||
command: /sbin/init
|
version: 2.7.14
|
||||||
steps:
|
steps:
|
||||||
# Machine Setup
|
# Machine Setup
|
||||||
# If you break your build into multiple jobs with workflows, you will probably want to do the parts of this that are relevant in each
|
# If you break your build into multiple jobs with workflows, you will probably want to do the parts of this that are relevant in each
|
||||||
@@ -42,25 +34,24 @@ jobs:
|
|||||||
# This is based on your 1.0 configuration file or project settings
|
# This is based on your 1.0 configuration file or project settings
|
||||||
- run:
|
- run:
|
||||||
working_directory: ~/rocky/python-uncompyle6
|
working_directory: ~/rocky/python-uncompyle6
|
||||||
command: pyenv local 2.7.11 && pyenv rehash && pip install virtualenv && pip install nose && pip install pep8 && pip install six && pyenv rehash
|
command: pip install virtualenv && pip install nose && pip install pep8 && pyenv rehash
|
||||||
# Dependencies
|
# Dependencies
|
||||||
# This would typically go in either a build or a build-and-test job when using workflows
|
# This would typically go in either a build or a build-and-test job when using workflows
|
||||||
# Restore the dependency cache
|
# Restore the dependency cache
|
||||||
- restore_cache:
|
- restore_cache:
|
||||||
keys:
|
keys:
|
||||||
# This branch if available
|
- v2-dependencies-{{ .Branch }}-
|
||||||
- v1-dep-{{ .Branch }}-
|
# fallback to using the latest cache if no exact match is found
|
||||||
# Default branch if not
|
- v2-dependencies-
|
||||||
- v1-dep-master-
|
|
||||||
# Any branch if there are none on the default branch - this should be unnecessary if you have your default branch configured correctly
|
|
||||||
- v1-dep-
|
|
||||||
# This is based on your 1.0 configuration file or project settings
|
# This is based on your 1.0 configuration file or project settings
|
||||||
- run: pip install --upgrade setuptools
|
- run: pip install --upgrade setuptools
|
||||||
- run: pip install -e .
|
- run: pip install -e .
|
||||||
- run: pip install pytest==3.2.5 hypothesis==3.0.0
|
- run: pip install -r requirements-dev.txt
|
||||||
|
|
||||||
# Save dependency cache
|
# Save dependency cache
|
||||||
- save_cache:
|
- save_cache:
|
||||||
key: v1-dep-{{ .Branch }}-{{ epoch }}
|
key: v2-dependencies-{{ .Branch }}-{{ epoch }}
|
||||||
paths:
|
paths:
|
||||||
# This is a broad list of cache paths to include many possible development environments
|
# This is a broad list of cache paths to include many possible development environments
|
||||||
# You can probably delete some of these entries
|
# You can probably delete some of these entries
|
||||||
@@ -69,14 +60,13 @@ jobs:
|
|||||||
- ~/.m2
|
- ~/.m2
|
||||||
- ~/.ivy2
|
- ~/.ivy2
|
||||||
- ~/.bundle
|
- ~/.bundle
|
||||||
- ~/.go_workspace
|
|
||||||
- ~/.gradle
|
|
||||||
- ~/.cache/bower
|
- ~/.cache/bower
|
||||||
|
|
||||||
# Test
|
# Test
|
||||||
# This would typically be a build job when using workflows, possibly combined with build
|
# This would typically be a build job when using workflows, possibly combined with build
|
||||||
# This is based on your 1.0 configuration file or project settings
|
# This is based on your 1.0 configuration file or project settings
|
||||||
- run: python ./setup.py develop && make check-2.7
|
- run: python ./setup.py develop && make check-2.7
|
||||||
- run: cd ./test/stdlib && pyenv local 2.7.11 && bash ./runtests.sh 'test_[p-z]*.py'
|
- run: cd ./test/stdlib && bash ./runtests.sh 'test_[p-z]*.py'
|
||||||
# Teardown
|
# Teardown
|
||||||
# If you break your build into multiple jobs with workflows, you will probably want to do the parts of this that are relevant in each
|
# If you break your build into multiple jobs with workflows, you will probably want to do the parts of this that are relevant in each
|
||||||
# Save test results
|
# Save test results
|
||||||
|
1
.gitignore
vendored
1
.gitignore
vendored
@@ -6,6 +6,7 @@
|
|||||||
/.eggs
|
/.eggs
|
||||||
/.hypothesis
|
/.hypothesis
|
||||||
/.idea
|
/.idea
|
||||||
|
/.mypy_cache
|
||||||
/.pytest_cache
|
/.pytest_cache
|
||||||
/.python-version
|
/.python-version
|
||||||
/.tox
|
/.tox
|
||||||
|
@@ -3,9 +3,9 @@ language: python
|
|||||||
python:
|
python:
|
||||||
- '3.5'
|
- '3.5'
|
||||||
- '2.7'
|
- '2.7'
|
||||||
- '2.6'
|
|
||||||
- '3.4'
|
- '3.4'
|
||||||
- '3.6'
|
- '3.6'
|
||||||
|
- '3.8'
|
||||||
|
|
||||||
matrix:
|
matrix:
|
||||||
include:
|
include:
|
||||||
|
@@ -115,7 +115,7 @@ mechanisms and addressed problems and extensions by some other means.
|
|||||||
Specifically, in `uncompyle`, decompilation of python bytecode 2.5 &
|
Specifically, in `uncompyle`, decompilation of python bytecode 2.5 &
|
||||||
2.6 is done by transforming the byte code into a pseudo-2.7 Python
|
2.6 is done by transforming the byte code into a pseudo-2.7 Python
|
||||||
bytecode and is based on code from Eloi Vanderbeken. A bit of this
|
bytecode and is based on code from Eloi Vanderbeken. A bit of this
|
||||||
could have bene easily added by modifying grammar rules.
|
could have been easily added by modifying grammar rules.
|
||||||
|
|
||||||
This project, `uncompyle6`, abandons that approach for various
|
This project, `uncompyle6`, abandons that approach for various
|
||||||
reasons. Having a grammar per Python version is much cleaner and it
|
reasons. Having a grammar per Python version is much cleaner and it
|
||||||
|
11
Makefile
11
Makefile
@@ -40,12 +40,19 @@ check-3.0 check-3.1 check-3.2 check-3.6:
|
|||||||
check-3.7: pytest
|
check-3.7: pytest
|
||||||
$(MAKE) -C test check
|
$(MAKE) -C test check
|
||||||
|
|
||||||
|
check-3.8:
|
||||||
|
$(MAKE) -C test check
|
||||||
|
|
||||||
#:PyPy 2.6.1 PyPy 5.0.1, or PyPy 5.8.0-beta0
|
#:PyPy 2.6.1 PyPy 5.0.1, or PyPy 5.8.0-beta0
|
||||||
# Skip for now
|
# Skip for now
|
||||||
2.6 5.0 5.3 5.6 5.8:
|
2.6 5.0 5.3 5.6 5.8:
|
||||||
|
|
||||||
#:PyPy pypy3-2.4.0 Python 3:
|
#:PyPy pypy3-2.4.0 Python 3.6.1:
|
||||||
pypy-3.2 2.4:
|
7.1 pypy-3.2 2.4:
|
||||||
|
$(MAKE) -C test $@
|
||||||
|
|
||||||
|
#:PyPy pypy3-2.4.0 Python 3.6.9:
|
||||||
|
7.2:
|
||||||
$(MAKE) -C test $@
|
$(MAKE) -C test $@
|
||||||
|
|
||||||
#: Run py.test tests
|
#: Run py.test tests
|
||||||
|
336
NEWS.md
336
NEWS.md
@@ -1,3 +1,165 @@
|
|||||||
|
3.6.0: 2019-12-10 gecko gecko
|
||||||
|
=============================
|
||||||
|
|
||||||
|
The main focus in this release was more accurate decompilation especially
|
||||||
|
for 3.7 and 3.8. However there are some improvments to Python 2.x as well,
|
||||||
|
including one of the long-standing problems of detecting the difference between
|
||||||
|
`try ... ` and `try else ...`.
|
||||||
|
|
||||||
|
With this release we now rebase Python 3.7 on off of a 3.7 base; This
|
||||||
|
is also as it is (now) in decompyle3. This facilitates removing some of the
|
||||||
|
cruft in control-flow detection in the 2.7 uncompyle2 base.
|
||||||
|
|
||||||
|
Alas, decompilation speed for 3.7 on is greatly increased. Hopefull
|
||||||
|
this is temporary (cough, cough) until we can do a static control flow
|
||||||
|
pass.
|
||||||
|
|
||||||
|
Finally, runing in 3.9-dev is tolerated. We can disassemble, but no parse tables yet.
|
||||||
|
|
||||||
|
|
||||||
|
3.5.1 2019-11-17 JNC
|
||||||
|
====================
|
||||||
|
|
||||||
|
- Pypy 3.3, 3.5, 3.6, and 3.6.9 support
|
||||||
|
- bump xdis version to handle newer Python releases, e.g. 2.7.17, 3.5.8, and 3.5.9
|
||||||
|
- Improve 3.0 decompilation
|
||||||
|
- no parse errors on stlib bytecode. However accurate translation in
|
||||||
|
control-flow and and/or detection needs work
|
||||||
|
- Remove extraneous iter() in "for" of list comprehension Fixes #272
|
||||||
|
- "for" block without a `POP_BLOCK `and confusing `JUMP_BACK` for `CONTINUE`. Fixes #293
|
||||||
|
- Fix unmarshal incompletness detected in Pypy 3.6
|
||||||
|
- Miscellaneous bugs fixed
|
||||||
|
|
||||||
|
3.5.0 2019-10-12 Stony Brook Ride
|
||||||
|
=================================
|
||||||
|
|
||||||
|
- Fix fragment bugs
|
||||||
|
* missing `RETURN_LAST` introduced when adding transformation layer
|
||||||
|
* more parent entries on tokens
|
||||||
|
- Preliminary support for decompiling Python 1.0, 1.1, 1.2, and 1.6
|
||||||
|
* Newer _xdis_ version needed
|
||||||
|
|
||||||
|
3.4.1 2019-10-02
|
||||||
|
================
|
||||||
|
|
||||||
|
- Correct assert{,2} transforms Fixes #289
|
||||||
|
- Fragment parsing fixes:
|
||||||
|
* Wasn't handling 3-arg `%p`
|
||||||
|
* fielding error in `code_deparse()`
|
||||||
|
- Use newer _xdis_ to better track Python 3.8.0
|
||||||
|
|
||||||
|
|
||||||
|
3.4.0 2019-08-24 Totoro
|
||||||
|
=======================
|
||||||
|
|
||||||
|
The main change is to add a tree-transformation phase. This simplifies the
|
||||||
|
code a little and allows us to turn `if ...: raise AssertionError` into
|
||||||
|
`assert`, and many `if ..: else if ...` into `if ... elif ..`
|
||||||
|
|
||||||
|
Use options `--show=before` and `--show=after` to see the before the tree transformation phase and after the tree transformation phase.
|
||||||
|
|
||||||
|
Most of the heavy lifting for this was done by x0ret.
|
||||||
|
|
||||||
|
Other changes:
|
||||||
|
|
||||||
|
- Fix issue #275, #283 (process to fix this bug is documented on wiki), #284
|
||||||
|
- blacken more code
|
||||||
|
- CircleCI adjustments for a changing CircleCi
|
||||||
|
- Require more recent `xdis` for Python 3.8
|
||||||
|
- Fix bugs in code using `BUILD_LIST_UNPACK` and variants
|
||||||
|
|
||||||
|
3.3.5 2019-07-03 Pre Independence Day
|
||||||
|
=====================================
|
||||||
|
|
||||||
|
Again, most of the work in this is release is thanks to x0ret.
|
||||||
|
|
||||||
|
- Handle annotation arguments in Python 3.x
|
||||||
|
- Fix _vararg_ and function signatures in 3.x
|
||||||
|
- Some 3.x < 3.6 `while` (1)/`if` fixes — others remain
|
||||||
|
- Start reinstating `else if` -> `elif`
|
||||||
|
- `LOAD_CONST` -> `LOAD_CODE` where appropriate
|
||||||
|
- option `--weak-verify` is now `--syntax-verify`
|
||||||
|
- code cleanups, start using [black](https://github.com/python/black) to reformat text
|
||||||
|
|
||||||
|
|
||||||
|
3.3.4 2019-06-19 Fleetwood at 65
|
||||||
|
================================
|
||||||
|
|
||||||
|
Most of the work in this is release is thanks to x0ret.
|
||||||
|
|
||||||
|
- Major work was done by x0ret to correct function signatures and include annotation types
|
||||||
|
- Handle Python 3.6 `STORE_ANNOTATION` [#58](https://github.com/rocky/python-uncompyle6/issues/58)
|
||||||
|
- Friendlier assembly output
|
||||||
|
- `LOAD_CONST` replaced by `LOAD_STR` where appropriate to simplify parsing and improve clarity
|
||||||
|
- remove unneeded parenthesis in a generator expression when it is the single argument to the function [#247](https://github.com/rocky/python-uncompyle6/issues/246)
|
||||||
|
- Bug in noting an async function [#246](https://github.com/rocky/python-uncompyle6/issues/246)
|
||||||
|
- Handle Unicode docstrings and fix docstring bugs [#241](https://github.com/rocky/python-uncompyle6/issues/241)
|
||||||
|
- Add short option -T as an alternate for --tree+
|
||||||
|
- Some grammar cleanup
|
||||||
|
|
||||||
|
3.3.3 2019-05-19 Henry and Lewis
|
||||||
|
================================
|
||||||
|
|
||||||
|
As before, decomplation bugs fixed. The focus has primarily been on
|
||||||
|
Python 3.7. But with this release, releases will be put on hold,as a
|
||||||
|
better control-flow detection is worked on . This has been needed for a
|
||||||
|
while, and is long overdue. It will probably also take a while to get
|
||||||
|
done as good as what we have now.
|
||||||
|
|
||||||
|
However this work will be done in a new project
|
||||||
|
[decompyle3](https://github.com/rocky/python-decompile3). In contrast
|
||||||
|
to _uncompyle6_ the code will be written assuming a modern Python 3,
|
||||||
|
e.g. 3.7. It is originally intended to decompile Python version 3.7
|
||||||
|
and greater.
|
||||||
|
|
||||||
|
* A number of Python 3.7+ chained comparisons were fixed
|
||||||
|
* Revise Python 3.6ish format string handling
|
||||||
|
* Go over operator precedence, e.g. for AST `IfExp`
|
||||||
|
|
||||||
|
Reported Bug Fixes
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* [#239: 3.7 handling of 4-level attribute import](https://github.com/rocky/python-uncompyle6/issues/239),
|
||||||
|
* [#229: Inconsistent if block in python3.6](https://github.com/rocky/python-uncompyle6/issues/229),
|
||||||
|
* [#227: Args not appearing in decompiled src when kwargs is specified explicitly (call_ex_kw)](https://github.com/rocky/python-uncompyle6/issues/227)
|
||||||
|
2.7 confusion around "and" versus comprehension "if"
|
||||||
|
* [#225: 2.7 confusion around "and" vs comprehension "if"](https://github.com/rocky/python-uncompyle6/issues/225)
|
||||||
|
|
||||||
|
3.3.2 2019-05-03 Better Friday
|
||||||
|
==============================
|
||||||
|
|
||||||
|
As before, lots of decomplation bugs fixed. The focus has primarily
|
||||||
|
been on Python 3.6. We can now parse the entire 3.6.8 Python library
|
||||||
|
and verify that without an error. The same is true for 3.5.8. A number
|
||||||
|
of the bugs fixed though are not contained to these versions. In fact
|
||||||
|
some span back as far as 2.x
|
||||||
|
|
||||||
|
But as before, many more remain in the 3.7 and 3.8 range which will
|
||||||
|
get addressed in future releases
|
||||||
|
|
||||||
|
Pypy 3.6 support was started. Pypy 3.x detection fixed (via `xdis`)
|
||||||
|
|
||||||
|
3.3.1 2019-04-19 Good Friday
|
||||||
|
==========================
|
||||||
|
|
||||||
|
Lots of decomplation bugs, especially in the 3.x series fixed. Don't worry though, many more remain.
|
||||||
|
|
||||||
|
* Add annotation return values in 3.6+
|
||||||
|
* Fix 3.6+ lambda parameter handling decompilation
|
||||||
|
* Fix 3.7+ chained comparison decompilation
|
||||||
|
* split out semantic-action customization into more separate files
|
||||||
|
* Add 3.8 try/else
|
||||||
|
* Fix 2.7 generator decompilation
|
||||||
|
* Fix some parser failures fixes in 3.4+ using test_pyenvlib
|
||||||
|
* Add more run tests
|
||||||
|
|
||||||
|
3.3.0 2019-04-14 Holy Week
|
||||||
|
==========================
|
||||||
|
|
||||||
|
* First cut at Python 3.8 (many bugs remain)
|
||||||
|
* Reinstate -c | --compile (compile before disassembly) option
|
||||||
|
* The usual smattering of bug and doc fixes
|
||||||
|
|
||||||
3.2.6 2019-03-23 Mueller Report
|
3.2.6 2019-03-23 Mueller Report
|
||||||
=======================================
|
=======================================
|
||||||
|
|
||||||
@@ -6,23 +168,25 @@ Mostly more of the same: bug fixes and pull requests.
|
|||||||
Bug Fixes
|
Bug Fixes
|
||||||
-----------
|
-----------
|
||||||
|
|
||||||
* [#155: Python 3.x bytecode confusing "try/else" with "try" in a loop](https://github.com/rocky/python-uncompyle6/issues/155),
|
* [#221: Wrong grammar for nested ifelsestmt (in Python 3.7 at least)](https://github.com/rocky/python-uncompyle6/issues/221)
|
||||||
* [#200: Python 3 bug in not detecting end bounds of an "if" ... "elif"](https://github.com/rocky/python-uncompyle6/issues/200),
|
|
||||||
* [#208: Comma placement in 3.6 and 3.7 **kwargs](https://github.com/rocky/python-uncompyle6/issues/208),
|
|
||||||
* [#209: Fix "if" return boundary in 3.6+](https://github.com/rocky/python-uncompyle6/issues/209),
|
|
||||||
* [#215: 2.7 can have two JUMP_BACKs at the end of a while loop](https://github.com/rocky/python-uncompyle6/issues/215)
|
* [#215: 2.7 can have two JUMP_BACKs at the end of a while loop](https://github.com/rocky/python-uncompyle6/issues/215)
|
||||||
|
* [#209: Fix "if" return boundary in 3.6+](https://github.com/rocky/python-uncompyle6/issues/209),
|
||||||
|
* [#208: Comma placement in 3.6 and 3.7 **kwargs](https://github.com/rocky/python-uncompyle6/issues/208),
|
||||||
|
* [#200: Python 3 bug in not detecting end bounds of an "if" ... "elif"](https://github.com/rocky/python-uncompyle6/issues/200),
|
||||||
|
* [#155: Python 3.x bytecode confusing "try/else" with "try" in a loop](https://github.com/rocky/python-uncompyle6/issues/155),
|
||||||
|
|
||||||
|
|
||||||
Pull Requests
|
Pull Requests
|
||||||
----------------
|
----------------
|
||||||
|
|
||||||
* [#202: Better "assert" statement detemination in Python 2.7](https://github.com/rocky/python-uncompyle6/pull/211)
|
* [#202: Better "assert" statement determination in Python 2.7](https://github.com/rocky/python-uncompyle6/pull/211)
|
||||||
* [#204: Python 3.7 testing](https://github.com/rocky/python-uncompyle6/pull/204)
|
* [#204: Python 3.7 testing](https://github.com/rocky/python-uncompyle6/pull/204)
|
||||||
* [#205: Run more f-string tests on Python 3.7](https://github.com/rocky/python-uncompyle6/pull/205)
|
* [#205: Run more f-string tests on Python 3.7](https://github.com/rocky/python-uncompyle6/pull/205)
|
||||||
* [#211: support utf-8 chars in Python 3 sourcecode](https://github.com/rocky/python-uncompyle6/pull/202)
|
* [#211: support utf-8 chars in Python 3 sourcecode](https://github.com/rocky/python-uncompyle6/pull/202)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
3.2.5 2018-12-30 Clearout sale
|
3.2.5 2018-12-30 Clear-out sale
|
||||||
======================================
|
======================================
|
||||||
|
|
||||||
- 3.7.2 Remove deprecation warning on regexp string that isn't raw
|
- 3.7.2 Remove deprecation warning on regexp string that isn't raw
|
||||||
@@ -57,7 +221,7 @@ Pull Requests
|
|||||||
|
|
||||||
- Add rudimentary 1.4 support (still a bit buggy)
|
- Add rudimentary 1.4 support (still a bit buggy)
|
||||||
- add --tree+ option to show formatting rule, when it is constant
|
- add --tree+ option to show formatting rule, when it is constant
|
||||||
- Python 2.7.15candidate1 support (via xdis)
|
- Python 2.7.15candidate1 support (via `xdis`)
|
||||||
- bug fixes, especially for 3.7 (but 2.7 and 3.6 and others as well)
|
- bug fixes, especially for 3.7 (but 2.7 and 3.6 and others as well)
|
||||||
|
|
||||||
3.1.3 2018-04-16
|
3.1.3 2018-04-16
|
||||||
@@ -87,14 +251,14 @@ Jesus on Friday's New York Times puzzle: "I'm stuck on 2A"
|
|||||||
- reduce 3.5, 3.6 control-flow bugs
|
- reduce 3.5, 3.6 control-flow bugs
|
||||||
- reduce ambiguity in rules that lead to long (exponential?) parses
|
- reduce ambiguity in rules that lead to long (exponential?) parses
|
||||||
- limit/isolate some 2.6/2.7,3.x grammar rules
|
- limit/isolate some 2.6/2.7,3.x grammar rules
|
||||||
- more runtime testing of decompiled code
|
- more run-time testing of decompiled code
|
||||||
- more removal of parenthesis around calls via setting precidence
|
- more removal of parenthesis around calls via setting precedence
|
||||||
|
|
||||||
3.1.0 2018-03-21 Equinox
|
3.1.0 2018-03-21 Equinox
|
||||||
==============================
|
==============================
|
||||||
|
|
||||||
- Add code_deparse_with_offset() fragment function.
|
- Add code_deparse_with_offset() fragment function.
|
||||||
- Correct paramenter call fragment deparse_code()
|
- Correct parameter call fragment deparse_code()
|
||||||
- Lots of 3.6, 3.x, and 2.7 bug fixes
|
- Lots of 3.6, 3.x, and 2.7 bug fixes
|
||||||
About 5% of 3.6 fail parsing now. But
|
About 5% of 3.6 fail parsing now. But
|
||||||
semantics still needs much to be desired.
|
semantics still needs much to be desired.
|
||||||
@@ -162,24 +326,24 @@ function calls and definitions.
|
|||||||
2.15.1 2018-01-27
|
2.15.1 2018-01-27
|
||||||
=====================
|
=====================
|
||||||
|
|
||||||
- Add --linemap option to give line correspondences
|
- Add `--linemap` option to give line correspondences
|
||||||
between original source lines and reconstructed line sources.
|
between original source lines and reconstructed line sources.
|
||||||
It is far from perfect, but it is a start
|
It is far from perfect, but it is a start
|
||||||
- Add a new class of tests: tests which when decompiled check themselves
|
- Add a new class of tests: tests which when decompiled check themselves
|
||||||
- Split off Python version semantic action customizations into its own file
|
- Split off Python version semantic action customizations into its own file
|
||||||
- Fix 2.7 bug in ifelse loop statement
|
- Fix 2.7 bug in `if`/`else` loop statement
|
||||||
- Handle 3.6+ EXTENDED_ARGs for POP_JUMP_IF... instructions
|
- Handle 3.6+ `EXTENDED_ARG`s for `POP_JUMP_IF..` instructions
|
||||||
- Correct 3.6+ calls with kwargs
|
- Correct 3.6+ calls with `kwargs`
|
||||||
- Describe the difficulty of 3.6 in README
|
- Describe the difficulty of 3.6 in README
|
||||||
|
|
||||||
2.14.3 2018-01-19
|
2.14.3 2018-01-19
|
||||||
=====================
|
=====================
|
||||||
|
|
||||||
- Fix bug in 3.5+ await stmt
|
- Fix bug in 3.5+ `await` statement
|
||||||
- Better version to magic handling; handle 3.5.2 .. 3.5.4 versions
|
- Better version to magic handling; handle 3.5.2 .. 3.5.4 versions
|
||||||
- Improve/correct test_pyenvlib.py status messages
|
- Improve/correct test_pyenvlib.py status messages
|
||||||
- Fix some 2.7 and 2.6 parser bugs
|
- Fix some 2.7 and 2.6 parser bugs
|
||||||
- Fix whilelse parsing bugs
|
- Fix `whilelse` parsing bugs
|
||||||
- Correct 2.5- decorator parsing
|
- Correct 2.5- decorator parsing
|
||||||
- grammar for decorators matches AST a little more
|
- grammar for decorators matches AST a little more
|
||||||
- better tests in setup.py for running the right version of Python
|
- better tests in setup.py for running the right version of Python
|
||||||
@@ -190,15 +354,15 @@ function calls and definitions.
|
|||||||
|
|
||||||
Decompilation bug fixes, mostly 3.6 and pre 2.7
|
Decompilation bug fixes, mostly 3.6 and pre 2.7
|
||||||
|
|
||||||
- 3.6 FUNCTION_EX (somewhat)
|
- 3.6 `FUNCTION_EX` (somewhat)
|
||||||
- 3.6 FUNCTION_EX_KW fixes
|
- 3.6 `FUNCTION_EX_KW` fixes
|
||||||
- 3.6 MAKE_FUNCTION fixes
|
- 3.6 `MAKE_FUNCTION` fixes
|
||||||
- correct 3.5 CALL_FUNCTION_VAR
|
- correct 3.5 `CALL_FUNCTION_VAR`
|
||||||
- stronger 3.x "while 1" testing
|
- stronger 3.x "while 1" testing
|
||||||
- Fix bug in if's with "pass" bodies. Fixes #104
|
- Fix bug in if's with "pass" bodies. Fixes #104
|
||||||
- try/else and try/finally fixes on 2.6-
|
- try/else and try/finally fixes on 2.6-
|
||||||
- limit pypy customization to pypy
|
- limit pypy customization to pypy
|
||||||
- Add addr fields in COME_FROMS
|
- Add addr fields in `COME_FROM`S
|
||||||
- Allow use of full instructions in parser reduction routines
|
- Allow use of full instructions in parser reduction routines
|
||||||
- Reduce grammar in Python 3 by specialization more to specific
|
- Reduce grammar in Python 3 by specialization more to specific
|
||||||
Python versions
|
Python versions
|
||||||
@@ -207,11 +371,11 @@ Decompilation bug fixes, mostly 3.6 and pre 2.7
|
|||||||
2.14.1 2017-12-10 Dr. Gecko
|
2.14.1 2017-12-10 Dr. Gecko
|
||||||
===================================
|
===================================
|
||||||
|
|
||||||
- Many decompilation bugfixes
|
- Many decompilation bug fixes
|
||||||
- Grammar rule reduction and version isolation
|
- Grammar rule reduction and version isolation
|
||||||
- Match higher-level nonterminal names more closely
|
- Match higher-level nonterminal names more closely
|
||||||
with Python AST
|
with Python AST
|
||||||
- Start automated Python stdlib testing - full round trip
|
- Start automated Python _stdlib_ testing — full round trip
|
||||||
|
|
||||||
2.14.0 2017-11-26 johnnybamazing
|
2.14.0 2017-11-26 johnnybamazing
|
||||||
=========================================
|
=========================================
|
||||||
@@ -220,7 +384,7 @@ Decompilation bug fixes, mostly 3.6 and pre 2.7
|
|||||||
and remove used grammar rules
|
and remove used grammar rules
|
||||||
- Fix a number of bytecode decompile problems
|
- Fix a number of bytecode decompile problems
|
||||||
(many more remain)
|
(many more remain)
|
||||||
- Add stdlib/runtests.sh for even more rigorous testing
|
- Add `stdlib/runtests.sh` for even more rigorous testing
|
||||||
|
|
||||||
2.13.3 2017-11-13
|
2.13.3 2017-11-13
|
||||||
=====================
|
=====================
|
||||||
@@ -236,16 +400,16 @@ Overall: better 3.6 decompiling and some much needed code refactoring and cleanu
|
|||||||
rather trying to parse the bytecode array. This largely been done in for versions 3.x;
|
rather trying to parse the bytecode array. This largely been done in for versions 3.x;
|
||||||
3.0 custom mangling code has been reduced;
|
3.0 custom mangling code has been reduced;
|
||||||
some 2.x conversion has been done, but more is desired. This make it possible to...
|
some 2.x conversion has been done, but more is desired. This make it possible to...
|
||||||
- Handle EXTENDED_ARGS better. While relevant to all Python versions it is most noticeable in
|
- Handle `EXTENDED_ARGS` better. While relevant to all Python versions it is most noticeable in
|
||||||
version 3.6+ where in switching to wordcodes the size of operands has been reduced from 2**16
|
version 3.6+ where in switching to wordcodes the size of operands has been reduced from 2^16
|
||||||
to 2**8. JUMP instruction then often need EXTENDED_ARGS.
|
to 2^8. `JUMP` instruction then often need EXTENDED_ARGS.
|
||||||
- Refactor find_jump_targets() with via working of of instructions rather the bytecode array.
|
- Refactor find_jump_targets() with via working of of instructions rather the bytecode array.
|
||||||
- use --weak-verify more and additional fuzzing on verify()
|
- use `--weak-verify` more and additional fuzzing on verify()
|
||||||
- fragment parser now ignores errors in nested function definitions; an parameter was
|
- fragment parser now ignores errors in nested function definitions; an parameter was
|
||||||
added to assist here. Ignoring errors may be okay because the fragment parser often just needs,
|
added to assist here. Ignoring errors may be okay because the fragment parser often just needs,
|
||||||
well, *fragments*.
|
well, *fragments*.
|
||||||
- Distinguish RETURN_VALUE from RETURN_END_IF in exception bodies better in 3.6
|
- Distinguish `RETURN_VALUE` from `RETURN_END_IF` in exception bodies better in 3.6
|
||||||
- bug in 3.x language changes: import queue via import Queue
|
- bug in 3.x language changes: import queue via `import Queue`
|
||||||
- reinstate some bytecode tests since decompiling has gotten better
|
- reinstate some bytecode tests since decompiling has gotten better
|
||||||
- Revise how to report a bug
|
- Revise how to report a bug
|
||||||
|
|
||||||
@@ -265,12 +429,12 @@ Overall: better 3.6 decompiling and some much needed code refactoring and cleanu
|
|||||||
- Fixes in deparsing lambda expressions
|
- Fixes in deparsing lambda expressions
|
||||||
- Improve table-semantics descriptions
|
- Improve table-semantics descriptions
|
||||||
- Document hacky customize arg count better (until we can remove it)
|
- Document hacky customize arg count better (until we can remove it)
|
||||||
- Update to use xdis 3.7.0 or greater
|
- Update to use `xdis` 3.7.0 or greater
|
||||||
|
|
||||||
2.12.0 2017-09-26
|
2.12.0 2017-09-26
|
||||||
=====================
|
=====================
|
||||||
|
|
||||||
- Use xdis 3.6.0 or greater now
|
- Use `xdis` 3.6.0 or greater now
|
||||||
- Small semantic table cleanups
|
- Small semantic table cleanups
|
||||||
- Python 3.4's terms a little names better
|
- Python 3.4's terms a little names better
|
||||||
- Slightly more Python 3.7, but still failing a lot
|
- Slightly more Python 3.7, but still failing a lot
|
||||||
@@ -284,13 +448,13 @@ Overall: better 3.6 decompiling and some much needed code refactoring and cleanu
|
|||||||
2.11.4 2017-08-15
|
2.11.4 2017-08-15
|
||||||
=====================
|
=====================
|
||||||
|
|
||||||
* scanner and parser now allow 3-part version string lookups,
|
* scanner and parser now allow 3-part version string look ups,
|
||||||
e.g. 2.7.1 We allow a float here, but if passed a string like '2.7'. or
|
e.g. 2.7.1 We allow a float here, but if passed a string like '2.7'. or
|
||||||
* unpin 3.5.1. xdis 3.5.4 has been release and fixes the problems we had. Use that.
|
* unpin 3.5.1. `xdis` 3.5.4 has been release and fixes the problems we had. Use that.
|
||||||
* some routines here moved to xdis. Use the xdis version
|
* some routines here moved to `xdis`. Use the `xdis` version
|
||||||
* README.rst: Link typo Name is trepan2 now not trepan
|
* `README.rst`: Link typo Name is _trepan2_ now not _trepan_
|
||||||
* xdis-forced change adjust for COMPARE_OP "is-not" in
|
* xdis-forced change adjust for `COMPARE_OP` "is-not" in
|
||||||
semanatic routines. We need "is not".
|
semantic routines. We need "is not".
|
||||||
* Some PyPy tolerance in validate testing.
|
* Some PyPy tolerance in validate testing.
|
||||||
* Some pyston tolerance
|
* Some pyston tolerance
|
||||||
|
|
||||||
@@ -300,15 +464,15 @@ Overall: better 3.6 decompiling and some much needed code refactoring and cleanu
|
|||||||
Very minor changes
|
Very minor changes
|
||||||
|
|
||||||
- RsT doc fixes and updates
|
- RsT doc fixes and updates
|
||||||
- use newer xdis, but not too new; 3.5.2 breaks uncompyle6
|
- use newer `xdis`, but not too new; 3.5.2 breaks uncompyle6
|
||||||
- use xdis opcode sets
|
- use `xdis` opcode sets
|
||||||
- xdis "exception match" is now "exception-match"
|
- `xdis` "exception match" is now "exception-match"
|
||||||
|
|
||||||
2.11.2 2017-07-09
|
2.11.2 2017-07-09
|
||||||
=====================
|
=====================
|
||||||
|
|
||||||
- Start supporting Pypy 3.5 (5.7.1-beta)
|
- Start supporting Pypy 3.5 (5.7.1-beta)
|
||||||
- use xdis 3.5.0's opcode sets and require xdis 3.5.0
|
- use `xdis` 3.5.0's opcode sets and require `xdis` 3.5.0
|
||||||
- Correct some Python 2.4-2.6 loop detection
|
- Correct some Python 2.4-2.6 loop detection
|
||||||
- guard against badly formatted bytecode
|
- guard against badly formatted bytecode
|
||||||
|
|
||||||
@@ -316,55 +480,55 @@ Very minor changes
|
|||||||
=====================
|
=====================
|
||||||
|
|
||||||
- Python 3.x annotation and function signature fixes
|
- Python 3.x annotation and function signature fixes
|
||||||
- Bump xdis version
|
- Bump `xdis` version
|
||||||
- Small pysource bug fixes
|
- Small `pysource.py` bug fixes
|
||||||
|
|
||||||
2.11.0 2017-06-18 Fleetwood
|
2.11.0 2017-06-18 Fleetwood
|
||||||
==================================
|
==================================
|
||||||
|
|
||||||
- Major improvements in fragment tracking
|
- Major improvements in fragment tracking
|
||||||
* Add nonterminal node in extractInfo
|
* Add nonterminal node in `extractInfo()`
|
||||||
* tag more offsets in expressions
|
* tag more offsets in expressions
|
||||||
* tag array subscripts
|
* tag array subscripts
|
||||||
* set YIELD value offset in a <yield> expr
|
* set `YIELD` value offset in a _yield expr_
|
||||||
* fix a long-standing bug in not adjusting final AST when melding other deparse ASTs
|
* fix a long-standing bug in not adjusting final AST when melding other deparse ASTs
|
||||||
- Fixes yet again for make_function node handling; document what's up here
|
- Fixes yet again for make_function node handling; document what's up here
|
||||||
- Fix bug in snowflake Python 3.5 *args kwargs
|
- Fix bug in snowflake Python 3.5 `*args`, `kwargs`
|
||||||
|
|
||||||
2.10.1 2017-06-3 Marylin Frankel
|
2.10.1 2017-06-3 Marylin Frankel
|
||||||
========================================
|
========================================
|
||||||
|
|
||||||
- fix some fragments parsing bugs
|
- fix some fragments parsing bugs
|
||||||
- was returning the wrong type sometimes in deparse_code_around_offset()
|
- was returning the wrong type sometimes in `deparse_code_around_offset()`
|
||||||
- capture function name in offsets
|
- capture function name in offsets
|
||||||
- track changes to ifelstrmtr node from pysource into fragments
|
- track changes to `ifelstrmtr` node from `pysource.py` into fragments
|
||||||
|
|
||||||
2.10.0 2017-05-30 Elaine Gordon
|
2.10.0 2017-05-30 Elaine Gordon
|
||||||
=======================================
|
=======================================
|
||||||
|
|
||||||
- Add fuzzy offset deparse look up
|
- Add fuzzy offset deparse look up
|
||||||
- 3.6 bug fixes
|
- 3.6 bug fixes
|
||||||
- fix EXTENDED_ARGS handling (and in 2.6 and others)
|
- fix `EXTENDED_ARGS` handling (and in 2.6 and others)
|
||||||
- semantic routine make_function fragments.py
|
- semantic routine make_function fragments.py
|
||||||
- MAKE_FUNCTION handling
|
- `MAKE_FUNCTION` handling
|
||||||
- CALL_FUNCTION_EX handling
|
- `CALL_FUNCTION_EX` handling
|
||||||
- async property on defs
|
- `async` property on `defs`
|
||||||
- support for CALL_FUNCTION_KW (moagstar)
|
- support for `CALL_FUNCTION_KW` (moagstar)
|
||||||
- 3.5+ UNMAP_PACK and BUILD_UNMAP_PACK handling
|
- 3.5+ `UNMAP_PACK` and` BUILD_UNMAP_PACK` handling
|
||||||
- 3.5 FUNCTION_VAR bug
|
- 3.5 FUNCTION_VAR bug
|
||||||
- 3.x pass statement insdie while True
|
- 3.x pass statement inside `while True`
|
||||||
- Improve 3.2 decompilation
|
- Improve 3.2 decompilation
|
||||||
- Fixed -o argument processing (grkov90)
|
- Fixed `-o` argument processing (grkov90)
|
||||||
- Reduce scope of LOAD_ASSERT as expr to 3.4+
|
- Reduce scope of LOAD_ASSERT as expr to 3.4+
|
||||||
- "await" statement fixes
|
- `await` statement fixes
|
||||||
- 2.3, 2.4 "if 1 .." fixes
|
- 2.3, 2.4 "if 1 .." fixes
|
||||||
- 3.x annotation fixes
|
- 3.x annotation fixes
|
||||||
|
|
||||||
2.9.11 2017-04-06
|
2.9.11 2017-04-06
|
||||||
=====================
|
=====================
|
||||||
|
|
||||||
- Better support for Python 3.5+ BUILD_MAP_UNPACK
|
- Better support for Python 3.5+ `BUILD_MAP_UNPACK`
|
||||||
- Start 3.6 CALL_FUNCTION_EX support
|
- Start 3.6 `CALL_FUNCTION_EX` support
|
||||||
- Many decompilation bug fixes. (Many more remain). See ChangeLog
|
- Many decompilation bug fixes. (Many more remain). See ChangeLog
|
||||||
|
|
||||||
2.9.10 2017-02-25
|
2.9.10 2017-02-25
|
||||||
@@ -372,7 +536,7 @@ Very minor changes
|
|||||||
|
|
||||||
- Python grammar rule fixes
|
- Python grammar rule fixes
|
||||||
- Add ability to get grammar coverage on runs
|
- Add ability to get grammar coverage on runs
|
||||||
- Handle Python 3.6 opcode BUILD_CONST_KEYMAP
|
- Handle Python 3.6 opcode `BUILD_CONST_KEYMAP`
|
||||||
|
|
||||||
2.9.9 2016-12-16
|
2.9.9 2016-12-16
|
||||||
|
|
||||||
@@ -388,13 +552,13 @@ Very minor changes
|
|||||||
====================
|
====================
|
||||||
|
|
||||||
- Better control-flow detection
|
- Better control-flow detection
|
||||||
- pseudo instruction THEN in 2.x
|
- pseudo instruction `THEN` in 2.x
|
||||||
to disambiguate if from and
|
to disambiguate if from and
|
||||||
- fix bug in --verify option
|
- fix bug in `--verify` option
|
||||||
- DRY (a little) control-flow detection
|
- DRY (a little) control-flow detection
|
||||||
- fix syntax in tuples with one element
|
- fix syntax in tuples with one element
|
||||||
- if AST rule inheritance in Python 2.5
|
- if AST rule inheritance in Python 2.5
|
||||||
- NAME_MODULE removal for Python <= 2.4
|
- `NAME_MODULE` removal for Python <= 2.4
|
||||||
- verify call fixes for Python <= 2.4
|
- verify call fixes for Python <= 2.4
|
||||||
- more Python lint
|
- more Python lint
|
||||||
|
|
||||||
@@ -405,9 +569,9 @@ Very minor changes
|
|||||||
- Some Python 3.6 bytecode to wordcode conversion fixes
|
- Some Python 3.6 bytecode to wordcode conversion fixes
|
||||||
- option -g: show start-end range when possible
|
- option -g: show start-end range when possible
|
||||||
- track print_docstring move to help (used in python 3.1)
|
- track print_docstring move to help (used in python 3.1)
|
||||||
- verify: allow RETURN_VALUE to match RETURN_END_IF
|
- verify: allow `RETURN_VALUE` to match `RETURN_END_IF`
|
||||||
- some 3.2 compatibility
|
- some 3.2 compatibility
|
||||||
- Better Python 3 control flow detection by adding Pseudo ELSE opcodes
|
- Better Python 3 control flow detection by adding Pseudo `ELSE` opcodes
|
||||||
|
|
||||||
2.9.6 2016-12-04
|
2.9.6 2016-12-04
|
||||||
====================
|
====================
|
||||||
@@ -423,7 +587,7 @@ Very minor changes
|
|||||||
|
|
||||||
- Correct MANIFEST.in
|
- Correct MANIFEST.in
|
||||||
- More AST grammar checking
|
- More AST grammar checking
|
||||||
- --linemapping option or linenumbers.line_number_mapping()
|
- `--linemapping` option or _linenumbers.line_number_mapping()_
|
||||||
Shows correspondence of lines between source
|
Shows correspondence of lines between source
|
||||||
and decompiled source
|
and decompiled source
|
||||||
- Some control flow adjustments in code for 2.x.
|
- Some control flow adjustments in code for 2.x.
|
||||||
@@ -443,7 +607,7 @@ Very minor changes
|
|||||||
* improper while 1 else
|
* improper while 1 else
|
||||||
* docstring indent
|
* docstring indent
|
||||||
* 3.3 default values in lambda expressions
|
* 3.3 default values in lambda expressions
|
||||||
* start 3.0 decompilation (needs newer xdis)
|
* start 3.0 decompilation (needs newer `xdis`)
|
||||||
- Start grammar misparse checking
|
- Start grammar misparse checking
|
||||||
|
|
||||||
|
|
||||||
@@ -457,12 +621,12 @@ Very minor changes
|
|||||||
2.9.3 2016-10-26
|
2.9.3 2016-10-26
|
||||||
====================
|
====================
|
||||||
|
|
||||||
Release forced by incompatibility change in xdis 3.2.0.
|
Release forced by incompatibility change in` xdis` 3.2.0.
|
||||||
|
|
||||||
- Python 3.1 bugs:
|
- Python 3.1 bugs:
|
||||||
* handle "with ... as"
|
* handle `with`... `as`
|
||||||
* handle "with"
|
* handle `with`
|
||||||
* Start handling def (...) -> yy (has bugs still)
|
* Start handling `def` (...) -> _yy_ (has bugs still)
|
||||||
|
|
||||||
- DRY Python 3.x via inheritance
|
- DRY Python 3.x via inheritance
|
||||||
- Python 3.6 work (from Daniel Bradburn)
|
- Python 3.6 work (from Daniel Bradburn)
|
||||||
@@ -488,12 +652,12 @@ Release forced by incompatibility change in xdis 3.2.0.
|
|||||||
2.9.0 2016-10-09
|
2.9.0 2016-10-09
|
||||||
====================
|
====================
|
||||||
|
|
||||||
- Use xdis 3.0.0 protocol load_module.
|
- Use `xdis` 3.0.0 protocol `load_module`.
|
||||||
this Forces change in requirements.txt and _pkg_info_.py
|
this Forces change in requirements.txt and _pkg_info_.py
|
||||||
- Start Python 1.5 decompiling; another round of work is needed to
|
- Start Python 1.5 decompiling; another round of work is needed to
|
||||||
remove bugs
|
remove bugs
|
||||||
- Simplify python 2.1 grammar
|
- Simplify python 2.1 grammar
|
||||||
- Fix bug with -t ... Wasn't showing source text when -t option was given
|
- Fix bug with `-t` ... Wasn't showing source text when `-t` option was given
|
||||||
- Fix 2.1-2.6 bug in list comprehension
|
- Fix 2.1-2.6 bug in list comprehension
|
||||||
|
|
||||||
2.8.4 2016-10-08
|
2.8.4 2016-10-08
|
||||||
@@ -502,8 +666,8 @@ Release forced by incompatibility change in xdis 3.2.0.
|
|||||||
- Python 3 disassembly bug fixes
|
- Python 3 disassembly bug fixes
|
||||||
- Python 3.6 fstring bug fixes (from moagstar)
|
- Python 3.6 fstring bug fixes (from moagstar)
|
||||||
- Python 2.1 disassembly
|
- Python 2.1 disassembly
|
||||||
- COME_FROM suffixes added in Python3
|
- `COME_FROM` suffixes added in Python3
|
||||||
- use .py extension in verification disassembly
|
- use `.py` extension in verification disassembly
|
||||||
|
|
||||||
2.8.3 2016-09-11 live from NYC!
|
2.8.3 2016-09-11 live from NYC!
|
||||||
=======================================
|
=======================================
|
||||||
@@ -548,8 +712,8 @@ control-flow structure detection is done.
|
|||||||
- Add Python 2.2 decompilation
|
- Add Python 2.2 decompilation
|
||||||
|
|
||||||
- Fix bugs
|
- Fix bugs
|
||||||
* PyPy LOOKUP_METHOD bug
|
* PyPy `LOOKUP_METHOD` bug
|
||||||
* Python 3.6 FORMAT_VALUE handles expressions now
|
* Python 3.6 `FORMAT_VALUE` handles expressions now
|
||||||
|
|
||||||
2.8.0 2016-08-03
|
2.8.0 2016-08-03
|
||||||
====================
|
====================
|
||||||
@@ -599,7 +763,7 @@ control-flow structure detection is done.
|
|||||||
====================
|
====================
|
||||||
|
|
||||||
- Improve Python 2.6 bytecode deparsing:
|
- Improve Python 2.6 bytecode deparsing:
|
||||||
stdlib now will deparse something
|
_stdlib_ now will deparse something
|
||||||
- Better <2.6 vs. 2.7 grammar separation
|
- Better <2.6 vs. 2.7 grammar separation
|
||||||
- Fix some 2.7 deparsing bugs
|
- Fix some 2.7 deparsing bugs
|
||||||
- Fix bug in installing uncompyle6 script
|
- Fix bug in installing uncompyle6 script
|
||||||
@@ -662,9 +826,9 @@ control-flow structure detection is done.
|
|||||||
2.3.2 2016-05-1
|
2.3.2 2016-05-1
|
||||||
===================
|
===================
|
||||||
|
|
||||||
- Add --version option standalone scripts
|
- Add `--version` option standalone scripts
|
||||||
- Correct License information in package
|
- Correct License information in package
|
||||||
- expose fns uncompyle_file, load_file, and load_module
|
- expose functions `uncompyle_file()`, `load_file()`, and `load_module()`
|
||||||
- Start to DRY Python2 and Python3 grammars Separate out 3.2, and 3.5+
|
- Start to DRY Python2 and Python3 grammars Separate out 3.2, and 3.5+
|
||||||
specific grammar code
|
specific grammar code
|
||||||
- Fix bug in 3.5+ constant map parsing
|
- Fix bug in 3.5+ constant map parsing
|
||||||
@@ -672,12 +836,12 @@ control-flow structure detection is done.
|
|||||||
2.3.0, 2.3.1 2016-04-30
|
2.3.0, 2.3.1 2016-04-30
|
||||||
=============================
|
=============================
|
||||||
|
|
||||||
- Require spark_parser >= 1.1.0
|
- Require `spark_parser` >= 1.1.0
|
||||||
|
|
||||||
2.2.0 2016-04-30
|
2.2.0 2016-04-30
|
||||||
====================
|
====================
|
||||||
|
|
||||||
- Spark is no longer here but pulled separate package spark_parse
|
- Spark is no longer here but pulled separate package [spark_parser](https://pypi.org/project/spark_parser/)
|
||||||
- Python 3 parsing fixes
|
- Python 3 parsing fixes
|
||||||
- More tests
|
- More tests
|
||||||
|
|
||||||
@@ -687,7 +851,7 @@ control-flow structure detection is done.
|
|||||||
- Support single-mode (in addition to exec-mode) compilation
|
- Support single-mode (in addition to exec-mode) compilation
|
||||||
- Start to DRY Python 2 and Python 3 grammars
|
- Start to DRY Python 2 and Python 3 grammars
|
||||||
- Fix bug in if else ternary construct
|
- Fix bug in if else ternary construct
|
||||||
- Fix bug in uncomplye6 -d and -r options (via lelicopter)
|
- Fix bug in uncomplye6 `-d` and `-r` options (via lelicopter)
|
||||||
|
|
||||||
|
|
||||||
2.1.3 2016-01-02
|
2.1.3 2016-01-02
|
||||||
@@ -695,7 +859,7 @@ control-flow structure detection is done.
|
|||||||
|
|
||||||
- Limited support for decompiling Python 3.5
|
- Limited support for decompiling Python 3.5
|
||||||
- Improve Python 3 class deparsing
|
- Improve Python 3 class deparsing
|
||||||
- Handle MAKE_CLOSURE opcode
|
- Handle `MAKE_CLOSURE` opcode
|
||||||
- Start to DRY opcode code.
|
- Start to DRY opcode code.
|
||||||
- increase test coverage
|
- increase test coverage
|
||||||
- fix misc small bugs and some improvements
|
- fix misc small bugs and some improvements
|
||||||
@@ -737,5 +901,5 @@ Changes from uncompyle2
|
|||||||
|
|
||||||
SPARK:
|
SPARK:
|
||||||
add option to show grammar rules applied
|
add option to show grammar rules applied
|
||||||
allow Python-style # comments in grammar
|
allow Python-style `#` comments in grammar
|
||||||
Runs on Python 3 and Python 2
|
Runs on Python 3 and Python 2
|
||||||
|
133
README.rst
133
README.rst
@@ -1,4 +1,6 @@
|
|||||||
|buildstatus| |Latest Version| |Supported Python Versions|
|
|buildstatus| |Pypi Installs| |Latest Version| |Supported Python Versions|
|
||||||
|
|
||||||
|
|packagestatus|
|
||||||
|
|
||||||
uncompyle6
|
uncompyle6
|
||||||
==========
|
==========
|
||||||
@@ -11,9 +13,9 @@ Introduction
|
|||||||
------------
|
------------
|
||||||
|
|
||||||
*uncompyle6* translates Python bytecode back into equivalent Python
|
*uncompyle6* translates Python bytecode back into equivalent Python
|
||||||
source code. It accepts bytecodes from Python version 1.3 to version
|
source code. It accepts bytecodes from Python version 1.0 to version
|
||||||
3.7, spanning over 22 years of Python releases. We include Dropbox's
|
3.8, spanning over 24 years of Python releases. We include Dropbox's
|
||||||
Python 2.5 bytecode and some PyPy bytecode.
|
Python 2.5 bytecode and some PyPy bytecodes.
|
||||||
|
|
||||||
Why this?
|
Why this?
|
||||||
---------
|
---------
|
||||||
@@ -44,14 +46,15 @@ not exist and there is just bytecode. Again, my debuggers make use of
|
|||||||
this.
|
this.
|
||||||
|
|
||||||
There were (and still are) a number of decompyle, uncompyle,
|
There were (and still are) a number of decompyle, uncompyle,
|
||||||
uncompyle2, uncompyle3 forks around. Almost all of them come basically
|
uncompyle2, uncompyle3 forks around. Many of them come basically from
|
||||||
from the same code base, and (almost?) all of them are no longer
|
the same code base, and (almost?) all of them are no longer actively
|
||||||
actively maintained. One was really good at decompiling Python 1.5-2.3
|
maintained. One was really good at decompiling Python 1.5-2.3, another
|
||||||
or so, another really good at Python 2.7, but that only. Another
|
really good at Python 2.7, but that only. Another handles Python 3.2
|
||||||
handles Python 3.2 only; another patched that and handled only 3.3.
|
only; another patched that and handled only 3.3. You get the
|
||||||
You get the idea. This code pulls all of these forks together and
|
idea. This code pulls all of these forks together and *moves
|
||||||
*moves forward*. There is some serious refactoring and cleanup in this
|
forward*. There is some serious refactoring and cleanup in this code
|
||||||
code base over those old forks.
|
base over those old forks. Even more experimental refactoring is going
|
||||||
|
on in decompile3_.
|
||||||
|
|
||||||
This demonstrably does the best in decompiling Python across all
|
This demonstrably does the best in decompiling Python across all
|
||||||
Python versions. And even when there is another project that only
|
Python versions. And even when there is another project that only
|
||||||
@@ -73,11 +76,11 @@ fixed in the other decompilers.
|
|||||||
Requirements
|
Requirements
|
||||||
------------
|
------------
|
||||||
|
|
||||||
The code here can be run on Python versions 2.6 or later, PyPy 3-2.4,
|
The code here can be run on Python versions 2.6 or later, PyPy 3-2.4
|
||||||
or PyPy-5.0.1. Python versions 2.4-2.7 are supported in the
|
and later. Python versions 2.4-2.7 are supported in the python-2.4
|
||||||
python-2.4 branch. The bytecode files it can read have been tested on
|
branch. The bytecode files it can read have been tested on Python
|
||||||
Python bytecodes from versions 1.4, 2.1-2.7, and 3.0-3.6 and the
|
bytecodes from versions 1.4, 2.1-2.7, and 3.0-3.8 and later PyPy
|
||||||
above-mentioned PyPy versions.
|
versions.
|
||||||
|
|
||||||
Installation
|
Installation
|
||||||
------------
|
------------
|
||||||
@@ -86,15 +89,15 @@ This uses setup.py, so it follows the standard Python routine:
|
|||||||
|
|
||||||
::
|
::
|
||||||
|
|
||||||
pip install -e . # set up to run from source tree
|
$ pip install -e . # set up to run from source tree
|
||||||
# Or if you want to install instead
|
# Or if you want to install instead
|
||||||
python setup.py install # may need sudo
|
$ python setup.py install # may need sudo
|
||||||
|
|
||||||
A GNU makefile is also provided so :code:`make install` (possibly as root or
|
A GNU makefile is also provided so :code:`make install` (possibly as root or
|
||||||
sudo) will do the steps above.
|
sudo) will do the steps above.
|
||||||
|
|
||||||
Testing
|
Running Tests
|
||||||
-------
|
-------------
|
||||||
|
|
||||||
::
|
::
|
||||||
|
|
||||||
@@ -122,16 +125,32 @@ For usage help:
|
|||||||
|
|
||||||
$ uncompyle6 -h
|
$ uncompyle6 -h
|
||||||
|
|
||||||
If you want strong verification of the correctness of the
|
Verification
|
||||||
decompilation process, add the `--verify` option. But there are
|
------------
|
||||||
situations where this will indicate a failure, although the generated
|
|
||||||
program is semantically equivalent. Using option `--weak-verify` will
|
|
||||||
tell you if there is something definitely wrong. Generally, large
|
|
||||||
swaths of code are decompiled correctly, if not the entire program.
|
|
||||||
|
|
||||||
You can also cross compare the results with pycdc_ . Since they work
|
In older versions of Python it was possible to verify bytecode by
|
||||||
differently, bugs here often aren't in that, and vice versa.
|
decompiling bytecode, and then compiling using the Python interpreter
|
||||||
|
for that bytecode version. Having done this the bytecode produced
|
||||||
|
could be compared with the original bytecode. However as Python's code
|
||||||
|
generation got better, this no longer was feasible.
|
||||||
|
|
||||||
|
If you want Python syntax verification of the correctness of the
|
||||||
|
decompilation process, add the :code:`--syntax-verify` option. However since
|
||||||
|
Python syntax changes, you should use this option if the bytecode is
|
||||||
|
the right bytecode for the Python interpreter that will be checking
|
||||||
|
the syntax.
|
||||||
|
|
||||||
|
You can also cross compare the results with another python decompiler
|
||||||
|
like pycdc_ . Since they work differently, bugs here often aren't in
|
||||||
|
that, and vice versa.
|
||||||
|
|
||||||
|
There is an interesting class of these programs that is readily
|
||||||
|
available give stronger verification: those programs that when run
|
||||||
|
test themselves. Our test suite includes these.
|
||||||
|
|
||||||
|
And Python comes with another a set of programs like this: its test
|
||||||
|
suite for the standard library. We have some code in :code:`test/stdlib` to
|
||||||
|
facilitate this kind of checking too.
|
||||||
|
|
||||||
Known Bugs/Restrictions
|
Known Bugs/Restrictions
|
||||||
-----------------------
|
-----------------------
|
||||||
@@ -146,27 +165,6 @@ All of the Python decompilers that I have looked at have problems
|
|||||||
decompiling Python's control flow. In some cases we can detect an
|
decompiling Python's control flow. In some cases we can detect an
|
||||||
erroneous decompilation and report that.
|
erroneous decompilation and report that.
|
||||||
|
|
||||||
In older versions of Python it was possible to verify bytecode by
|
|
||||||
decompiling bytecode, and then compiling using the Python interpreter
|
|
||||||
for that bytecode version. Having done this the bytecode produced
|
|
||||||
could be compared with the original bytecode. However as Python's code
|
|
||||||
generation got better, this is no longer feasible.
|
|
||||||
|
|
||||||
There verification that we use that doesn't check bytecode for
|
|
||||||
equivalence but does check to see if the resulting decompiled source
|
|
||||||
is a valid Python program by running the Python interpreter. Because
|
|
||||||
the Python language has changed so much, for best results you should
|
|
||||||
use the same Python version in checking as was used in creating the
|
|
||||||
bytecode.
|
|
||||||
|
|
||||||
There are however an interesting class of these programs that is
|
|
||||||
readily available give stronger verification: those programs that
|
|
||||||
when run check some computation, or even better themselves.
|
|
||||||
|
|
||||||
And already Python has a set of programs like this: the test suite
|
|
||||||
for the standard library that comes with Python. We have some
|
|
||||||
code in `test/stdlib` to facilitate this kind of checking.
|
|
||||||
|
|
||||||
Python support is strongest in Python 2 for 2.7 and drops off as you
|
Python support is strongest in Python 2 for 2.7 and drops off as you
|
||||||
get further away from that. Support is also probably pretty good for
|
get further away from that. Support is also probably pretty good for
|
||||||
python 2.3-2.4 since a lot of the goodness of early the version of the
|
python 2.3-2.4 since a lot of the goodness of early the version of the
|
||||||
@@ -182,20 +180,30 @@ In the Python 3 series, Python support is is strongest around 3.4 or
|
|||||||
3.0 is weird in that it in some ways resembles 2.6 more than it does
|
3.0 is weird in that it in some ways resembles 2.6 more than it does
|
||||||
3.1 or 2.7. Python 3.6 changes things drastically by using word codes
|
3.1 or 2.7. Python 3.6 changes things drastically by using word codes
|
||||||
rather than byte codes. As a result, the jump offset field in a jump
|
rather than byte codes. As a result, the jump offset field in a jump
|
||||||
instruction argument has been reduced. This makes the `EXTENDED_ARG`
|
instruction argument has been reduced. This makes the :code:`EXTENDED_ARG`
|
||||||
instructions are now more prevalent in jump instruction; previously
|
instructions are now more prevalent in jump instruction; previously
|
||||||
they had been rare. Perhaps to compensate for the additional
|
they had been rare. Perhaps to compensate for the additional
|
||||||
`EXTENDED_ARG` instructions, additional jump optimization has been
|
:code:`EXTENDED_ARG` instructions, additional jump optimization has been
|
||||||
added. So in sum handling control flow by ad hoc means as is currently
|
added. So in sum handling control flow by ad hoc means as is currently
|
||||||
done is worse.
|
done is worse.
|
||||||
|
|
||||||
Between Python 3.5, 3.6 and 3.7 there have been major changes to the
|
Between Python 3.5, 3.6, 3.7 there have been major changes to the
|
||||||
`MAKE_FUNCTION` and `CALL_FUNCTION` instructions.
|
:code:`MAKE_FUNCTION` and :code:`CALL_FUNCTION` instructions. Python
|
||||||
|
|
||||||
|
Python 3.8 removes :code:`SETUP_LOOP`, :code:`SETUP_EXCEPT`,
|
||||||
|
:code:`BREAK_LOOP`, and :code:`CONTINUE_LOOP`, instructions which may
|
||||||
|
make control-flow detection harder, lacking the more sophisticated
|
||||||
|
control-flow analysis that is planned. We'll see.
|
||||||
|
|
||||||
Currently not all Python magic numbers are supported. Specifically in
|
Currently not all Python magic numbers are supported. Specifically in
|
||||||
some versions of Python, notably Python 3.6, the magic number has
|
some versions of Python, notably Python 3.6, the magic number has
|
||||||
changes several times within a version. We support only the released
|
changes several times within a version.
|
||||||
magic. There are also customized Python interpreters, notably Dropbox,
|
|
||||||
|
**We support only released versions, not candidate versions.** Note
|
||||||
|
however that the magic of a released version is usually the same as
|
||||||
|
the *last* candidate version prior to release.
|
||||||
|
|
||||||
|
There are also customized Python interpreters, notably Dropbox,
|
||||||
which use their own magic and encrypt bytcode. With the exception of
|
which use their own magic and encrypt bytcode. With the exception of
|
||||||
the Dropbox's old Python 2.5 interpreter this kind of thing is not
|
the Dropbox's old Python 2.5 interpreter this kind of thing is not
|
||||||
handled.
|
handled.
|
||||||
@@ -215,29 +223,34 @@ There is lots to do, so please dig in and help.
|
|||||||
See Also
|
See Also
|
||||||
--------
|
--------
|
||||||
|
|
||||||
* https://github.com/zrax/pycdc : purports to support all versions of Python. It is written in C++ and is most accurate for Python versions around 2.7 and 3.3 when the code was more actively developed. Accuracy for more recent versions of Python 3 and early versions of Python are especially lacking. See its `issue tracker <https://github.com/zrax/pycdc/issues>`_ for details. Currently lightly maintained.
|
* https://github.com/rocky/python-decompile3 : Much smaller and more modern code, focusing on 3.7+. Changes in that will get migrated back ehre.
|
||||||
* https://code.google.com/archive/p/unpyc3/ : supports Python 3.2 only. The above projects use a different decompiling technique than what is used here. Currently unmaintained.
|
* https://code.google.com/archive/p/unpyc3/ : supports Python 3.2 only. The above projects use a different decompiling technique than what is used here. Currently unmaintained.
|
||||||
* https://github.com/figment/unpyc3/ : fork of above, but supports Python 3.3 only. Includes some fixes like supporting function annotations. Currently unmaintained.
|
* https://github.com/figment/unpyc3/ : fork of above, but supports Python 3.3 only. Includes some fixes like supporting function annotations. Currently unmaintained.
|
||||||
* https://github.com/wibiti/uncompyle2 : supports Python 2.7 only, but does that fairly well. There are situtations where `uncompyle6` results are incorrect while `uncompyle2` results are not, but more often uncompyle6 is correct when uncompyle2 is not. Because `uncompyle6` adheres to accuracy over idiomatic Python, `uncompyle2` can produce more natural-looking code when it is correct. Currently `uncompyle2` is lightly maintained. See its issue `tracker <https://github.com/wibiti/uncompyle2/issues>`_ for more details
|
* https://github.com/wibiti/uncompyle2 : supports Python 2.7 only, but does that fairly well. There are situations where :code:`uncompyle6` results are incorrect while :code:`uncompyle2` results are not, but more often uncompyle6 is correct when uncompyle2 is not. Because :code:`uncompyle6` adheres to accuracy over idiomatic Python, :code:`uncompyle2` can produce more natural-looking code when it is correct. Currently :code:`uncompyle2` is lightly maintained. See its issue `tracker <https://github.com/wibiti/uncompyle2/issues>`_ for more details
|
||||||
* `How to report a bug <https://github.com/rocky/python-uncompyle6/blob/master/HOW-TO-REPORT-A-BUG.md>`_
|
* `How to report a bug <https://github.com/rocky/python-uncompyle6/blob/master/HOW-TO-REPORT-A-BUG.md>`_
|
||||||
* The HISTORY_ file.
|
* The HISTORY_ file.
|
||||||
* https://github.com/rocky/python-xdis : Cross Python version disassembler
|
* https://github.com/rocky/python-xdis : Cross Python version disassembler
|
||||||
* https://github.com/rocky/python-xasm : Cross Python version assembler
|
* https://github.com/rocky/python-xasm : Cross Python version assembler
|
||||||
* https://github.com/rocky/python-uncompyle6/wiki : Wiki Documents which describe the code and aspects of it in more detail
|
* https://github.com/rocky/python-uncompyle6/wiki : Wiki Documents which describe the code and aspects of it in more detail
|
||||||
|
* https://github.com/zrax/pycdc : The README for this C++ code syas it aims to support all versions of Python. It is best for Python versions around 2.7 and 3.3 when the code was initially developed. Accuracy for current versions of Python3 and early versions of Python is lacking. Without major effort, it is unlikely it can be made to support current Python 3. See its `issue tracker <https://github.com/zrax/pycdc/issues>`_ for details. Currently lightly maintained.
|
||||||
|
|
||||||
|
|
||||||
.. _trepan: https://pypi.python.org/pypi/trepan2
|
.. _trepan: https://pypi.python.org/pypi/trepan2g
|
||||||
.. _compiler: https://pypi.python.org/pypi/spark_parser
|
.. _compiler: https://pypi.python.org/pypi/spark_parser
|
||||||
.. _HISTORY: https://github.com/rocky/python-uncompyle6/blob/master/HISTORY.md
|
.. _HISTORY: https://github.com/rocky/python-uncompyle6/blob/master/HISTORY.md
|
||||||
.. _debuggers: https://pypi.python.org/pypi/trepan3k
|
.. _debuggers: https://pypi.python.org/pypi/trepan3k
|
||||||
.. _remake: https://bashdb.sf.net/remake
|
.. _remake: https://bashdb.sf.net/remake
|
||||||
.. _pycdc: https://github.com/zrax/pycdc
|
.. _pycdc: https://github.com/zrax/pycdc
|
||||||
|
.. _decompile3: https://github.com/rocky/python-decompile3
|
||||||
.. _this: https://github.com/rocky/python-uncompyle6/wiki/Deparsing-technology-and-its-use-in-exact-location-reporting
|
.. _this: https://github.com/rocky/python-uncompyle6/wiki/Deparsing-technology-and-its-use-in-exact-location-reporting
|
||||||
.. |buildstatus| image:: https://travis-ci.org/rocky/python-uncompyle6.svg
|
.. |buildstatus| image:: https://travis-ci.org/rocky/python-uncompyle6.svg
|
||||||
:target: https://travis-ci.org/rocky/python-uncompyle6
|
:target: https://travis-ci.org/rocky/python-uncompyle6
|
||||||
|
.. |packagestatus| image:: https://repology.org/badge/vertical-allrepos/python:uncompyle6.svg
|
||||||
|
:target: https://repology.org/project/python:uncompyle6/versions
|
||||||
.. _PJOrion: http://www.koreanrandom.com/forum/topic/15280-pjorion-%D1%80%D0%B5%D0%B4%D0%B0%D0%BA%D1%82%D0%B8%D1%80%D0%BE%D0%B2%D0%B0%D0%BD%D0%B8%D0%B5-%D0%BA%D0%BE%D0%BC%D0%BF%D0%B8%D0%BB%D1%8F%D1%86%D0%B8%D1%8F-%D0%B4%D0%B5%D0%BA%D0%BE%D0%BC%D0%BF%D0%B8%D0%BB%D1%8F%D1%86%D0%B8%D1%8F-%D0%BE%D0%B1%D1%84
|
.. _PJOrion: http://www.koreanrandom.com/forum/topic/15280-pjorion-%D1%80%D0%B5%D0%B4%D0%B0%D0%BA%D1%82%D0%B8%D1%80%D0%BE%D0%B2%D0%B0%D0%BD%D0%B8%D0%B5-%D0%BA%D0%BE%D0%BC%D0%BF%D0%B8%D0%BB%D1%8F%D1%86%D0%B8%D1%8F-%D0%B4%D0%B5%D0%BA%D0%BE%D0%BC%D0%BF%D0%B8%D0%BB%D1%8F%D1%86%D0%B8%D1%8F-%D0%BE%D0%B1%D1%84
|
||||||
.. _Deobfuscator: https://github.com/extremecoders-re/PjOrion-Deobfuscator
|
.. _Deobfuscator: https://github.com/extremecoders-re/PjOrion-Deobfuscator
|
||||||
.. _Py2EXE: https://en.wikipedia.org/wiki/Py2exe
|
.. _Py2EXE: https://en.wikipedia.org/wiki/Py2exe
|
||||||
.. |Supported Python Versions| image:: https://img.shields.io/pypi/pyversions/uncompyle6.svg
|
.. |Supported Python Versions| image:: https://img.shields.io/pypi/pyversions/uncompyle6.svg
|
||||||
.. |Latest Version| image:: https://badge.fury.io/py/uncompyle6.svg
|
.. |Latest Version| image:: https://badge.fury.io/py/uncompyle6.svg
|
||||||
:target: https://badge.fury.io/py/uncompyle6
|
:target: https://badge.fury.io/py/uncompyle6
|
||||||
|
.. |Pypi Installs| image:: https://pepy.tech/badge/uncompyle6/month
|
||||||
|
@@ -23,48 +23,49 @@
|
|||||||
|
|
||||||
# Things that change more often go here.
|
# Things that change more often go here.
|
||||||
copyright = """
|
copyright = """
|
||||||
Copyright (C) 2015-2018 Rocky Bernstein <rb@dustyfeet.com>.
|
Copyright (C) 2015-2019 Rocky Bernstein <rb@dustyfeet.com>.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
classifiers = ['Development Status :: 5 - Production/Stable',
|
classifiers = ["Development Status :: 5 - Production/Stable",
|
||||||
'Intended Audience :: Developers',
|
"Intended Audience :: Developers",
|
||||||
'License :: OSI Approved :: GNU General Public License v3 (GPLv3)',
|
"License :: OSI Approved :: GNU General Public License v3 (GPLv3)",
|
||||||
'Operating System :: OS Independent',
|
"Operating System :: OS Independent",
|
||||||
'Programming Language :: Python',
|
"Programming Language :: Python",
|
||||||
'Programming Language :: Python :: 2.4',
|
"Programming Language :: Python :: 2.4",
|
||||||
'Programming Language :: Python :: 2.5',
|
"Programming Language :: Python :: 2.5",
|
||||||
'Programming Language :: Python :: 2.6',
|
"Programming Language :: Python :: 2.6",
|
||||||
'Programming Language :: Python :: 2.7',
|
"Programming Language :: Python :: 2.7",
|
||||||
'Programming Language :: Python :: 3.0',
|
"Programming Language :: Python :: 3.0",
|
||||||
'Programming Language :: Python :: 3.1',
|
"Programming Language :: Python :: 3.1",
|
||||||
'Programming Language :: Python :: 3.2',
|
"Programming Language :: Python :: 3.2",
|
||||||
'Programming Language :: Python :: 3.3',
|
"Programming Language :: Python :: 3.3",
|
||||||
'Programming Language :: Python :: 3.4',
|
"Programming Language :: Python :: 3.4",
|
||||||
'Programming Language :: Python :: 3.5',
|
"Programming Language :: Python :: 3.5",
|
||||||
'Programming Language :: Python :: 3.6',
|
"Programming Language :: Python :: 3.6",
|
||||||
'Programming Language :: Python :: 3.7',
|
"Programming Language :: Python :: 3.7",
|
||||||
'Topic :: Software Development :: Debuggers',
|
"Programming Language :: Python :: 3.8",
|
||||||
'Topic :: Software Development :: Libraries :: Python Modules',
|
"Topic :: Software Development :: Debuggers",
|
||||||
|
"Topic :: Software Development :: Libraries :: Python Modules",
|
||||||
]
|
]
|
||||||
|
|
||||||
# The rest in alphabetic order
|
# The rest in alphabetic order
|
||||||
author = "Rocky Bernstein, Hartmut Goebel, John Aycock, and others"
|
author = "Rocky Bernstein, Hartmut Goebel, John Aycock, and others"
|
||||||
author_email = "rb@dustyfeet.com"
|
author_email = "rb@dustyfeet.com"
|
||||||
entry_points = {
|
entry_points = {
|
||||||
'console_scripts': [
|
"console_scripts": [
|
||||||
'uncompyle6=uncompyle6.bin.uncompile:main_bin',
|
"uncompyle6=uncompyle6.bin.uncompile:main_bin",
|
||||||
'pydisassemble=uncompyle6.bin.pydisassemble:main',
|
"pydisassemble=uncompyle6.bin.pydisassemble:main",
|
||||||
]}
|
]}
|
||||||
ftp_url = None
|
ftp_url = None
|
||||||
install_requires = ['spark-parser >= 1.8.7, < 1.9.0',
|
install_requires = ["spark-parser >= 1.8.9, < 1.9.0",
|
||||||
'xdis >= 3.9.0, < 3.10.0']
|
"xdis >= 4.2.0, < 4.3.0"]
|
||||||
|
|
||||||
license = 'GPL3'
|
license = "GPL3"
|
||||||
mailing_list = 'python-debugger@googlegroups.com'
|
mailing_list = "python-debugger@googlegroups.com"
|
||||||
modname = 'uncompyle6'
|
modname = "uncompyle6"
|
||||||
py_modules = None
|
py_modules = None
|
||||||
short_desc = 'Python cross-version byte-code decompiler'
|
short_desc = "Python cross-version byte-code decompiler"
|
||||||
web = 'https://github.com/rocky/python-uncompyle6/'
|
web = "https://github.com/rocky/python-uncompyle6/"
|
||||||
|
|
||||||
# tracebacks in zip files are funky and not debuggable
|
# tracebacks in zip files are funky and not debuggable
|
||||||
zip_safe = True
|
zip_safe = True
|
||||||
@@ -81,5 +82,5 @@ def read(*rnames):
|
|||||||
return open(os.path.join(srcdir, *rnames)).read()
|
return open(os.path.join(srcdir, *rnames)).read()
|
||||||
|
|
||||||
# Get info from files; set: long_description and VERSION
|
# Get info from files; set: long_description and VERSION
|
||||||
long_description = ( read("README.rst") + '\n' )
|
long_description = ( read("README.rst") + "\n" )
|
||||||
exec(read('uncompyle6/version.py'))
|
exec(read("uncompyle6/version.py"))
|
||||||
|
@@ -56,18 +56,21 @@
|
|||||||
|
|
||||||
$ . ./admin-tools/make-dist-older.sh
|
$ . ./admin-tools/make-dist-older.sh
|
||||||
$ git tag release-python-2.4-$VERSION
|
$ git tag release-python-2.4-$VERSION
|
||||||
|
$ twine check dist/uncompyle6-$VERSION*
|
||||||
$ . ./admin-tools/make-dist-newer.sh
|
$ . ./admin-tools/make-dist-newer.sh
|
||||||
$ git tag release-$VERSION
|
$ twine check dist/uncompyle6-$VERSION*
|
||||||
|
|
||||||
# Upload single package and look at Rst Formating
|
# Upload single package and look at Rst Formating
|
||||||
|
|
||||||
|
$ twine check dist/uncompyle6-${VERSION}*
|
||||||
$ twine upload dist/uncompyle6-${VERSION}-py3.3.egg
|
$ twine upload dist/uncompyle6-${VERSION}-py3.3.egg
|
||||||
|
|
||||||
# Upload rest of versions
|
# Upload rest of versions
|
||||||
|
|
||||||
$ twine upload dist/uncompyle6-${VERSION}*
|
$ twine upload dist/uncompyle6-${VERSION}*
|
||||||
|
|
||||||
|
Goto https://github.com/rocky/python-uncompyle6/releases
|
||||||
|
|
||||||
# Push tags:
|
# Push tags:
|
||||||
|
|
||||||
$ git push --tags
|
$ git push --tags
|
||||||
|
@@ -5,4 +5,4 @@ if [[ $0 == ${BASH_SOURCE[0]} ]] ; then
|
|||||||
echo "This script should be *sourced* rather than run directly through bash"
|
echo "This script should be *sourced* rather than run directly through bash"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
export PYVERSIONS='3.2.6 3.6.8 3.7.2 2.6.9 3.3.7 2.7.15 3.2.6 3.1.5 3.4.8'
|
export PYVERSIONS='3.5.9 3.6.9 2.6.9 3.3.7 2.7.17 3.2.6 3.1.5 3.4.10 3.7.5'
|
||||||
|
@@ -6,4 +6,4 @@ if [[ $0 == ${BASH_SOURCE[0]} ]] ; then
|
|||||||
echo "This script should be *sourced* rather than run directly through bash"
|
echo "This script should be *sourced* rather than run directly through bash"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
export PYVERSIONS='2.4.6 2.5.6'
|
export PYVERSIONS='2.4.6 2.5.6 2.6.9'
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
PYTHON_VERSION=3.6.5
|
PYTHON_VERSION=3.7.5
|
||||||
|
|
||||||
# FIXME put some of the below in a common routine
|
# FIXME put some of the below in a common routine
|
||||||
function finish {
|
function finish {
|
||||||
|
@@ -61,7 +61,7 @@ build_script:
|
|||||||
|
|
||||||
test_script:
|
test_script:
|
||||||
# Run the project tests
|
# Run the project tests
|
||||||
- "%CMD_IN_ENV% python test/test_pyenvlib.py --native --weak-verify"
|
- "%CMD_IN_ENV% python test/test_pyenvlib.py --native --syntax-verify"
|
||||||
|
|
||||||
after_test:
|
after_test:
|
||||||
# If tests are successful, create binary packages for the project.
|
# If tests are successful, create binary packages for the project.
|
||||||
|
@@ -1,78 +0,0 @@
|
|||||||
import sys
|
|
||||||
from uncompyle6 import PYTHON3
|
|
||||||
if PYTHON3:
|
|
||||||
from io import StringIO
|
|
||||||
minint = -sys.maxsize-1
|
|
||||||
maxint = sys.maxsize
|
|
||||||
else:
|
|
||||||
from StringIO import StringIO
|
|
||||||
minint = -sys.maxint-1
|
|
||||||
maxint = sys.maxint
|
|
||||||
from uncompyle6.semantics.helper import print_docstring
|
|
||||||
|
|
||||||
class PrintFake():
|
|
||||||
def __init__(self):
|
|
||||||
self.pending_newlines = 0
|
|
||||||
self.f = StringIO()
|
|
||||||
|
|
||||||
def write(self, *data):
|
|
||||||
if (len(data) == 0) or (len(data) == 1 and data[0] == ''):
|
|
||||||
return
|
|
||||||
out = ''.join((str(j) for j in data))
|
|
||||||
n = 0
|
|
||||||
for i in out:
|
|
||||||
if i == '\n':
|
|
||||||
n += 1
|
|
||||||
if n == len(out):
|
|
||||||
self.pending_newlines = max(self.pending_newlines, n)
|
|
||||||
return
|
|
||||||
elif n:
|
|
||||||
self.pending_newlines = max(self.pending_newlines, n)
|
|
||||||
out = out[n:]
|
|
||||||
break
|
|
||||||
else:
|
|
||||||
break
|
|
||||||
|
|
||||||
if self.pending_newlines > 0:
|
|
||||||
self.f.write('\n'*self.pending_newlines)
|
|
||||||
self.pending_newlines = 0
|
|
||||||
|
|
||||||
for i in out[::-1]:
|
|
||||||
if i == '\n':
|
|
||||||
self.pending_newlines += 1
|
|
||||||
else:
|
|
||||||
break
|
|
||||||
|
|
||||||
if self.pending_newlines:
|
|
||||||
out = out[:-self.pending_newlines]
|
|
||||||
self.f.write(out)
|
|
||||||
def println(self, *data):
|
|
||||||
if data and not(len(data) == 1 and data[0] ==''):
|
|
||||||
self.write(*data)
|
|
||||||
self.pending_newlines = max(self.pending_newlines, 1)
|
|
||||||
return
|
|
||||||
pass
|
|
||||||
|
|
||||||
def test_docstring():
|
|
||||||
|
|
||||||
for doc, expect in (
|
|
||||||
("Now is the time",
|
|
||||||
' """Now is the time"""'),
|
|
||||||
("""
|
|
||||||
Now is the time
|
|
||||||
""",
|
|
||||||
''' """
|
|
||||||
Now is the time
|
|
||||||
"""''')
|
|
||||||
|
|
||||||
# (r'''func placeholder - ' and with ("""\nstring\n """)''',
|
|
||||||
# """ r'''func placeholder - ' and with (\"\"\"\nstring\n\"\"\")'''"""),
|
|
||||||
# (r"""func placeholder - ' and with ('''\nstring\n''') and \"\"\"\nstring\n\"\"\" """,
|
|
||||||
# """ r\"\"\"func placeholder - ' and with ('''\nstring\n''') and \"\"\"\nstring\n\"\"\" \"\"\"""")
|
|
||||||
):
|
|
||||||
|
|
||||||
o = PrintFake()
|
|
||||||
# print(doc)
|
|
||||||
# print(expect)
|
|
||||||
print_docstring(o, ' ', doc)
|
|
||||||
assert expect == o.f.getvalue()
|
|
@@ -1,9 +1,12 @@
|
|||||||
# std
|
# std
|
||||||
# test
|
# test
|
||||||
from uncompyle6 import PYTHON_VERSION, deparse_code
|
import sys
|
||||||
|
from uncompyle6 import PYTHON_VERSION, code_deparse
|
||||||
import pytest
|
import pytest
|
||||||
pytestmark = pytest.mark.skipif(PYTHON_VERSION <= 2.6,
|
|
||||||
reason='hypothesis needs 2.7 or later')
|
pytestmark = pytest.mark.skipif(
|
||||||
|
PYTHON_VERSION <= 2.6, reason="hypothesis needs 2.7 or later"
|
||||||
|
)
|
||||||
if PYTHON_VERSION > 2.6:
|
if PYTHON_VERSION > 2.6:
|
||||||
|
|
||||||
import hypothesis
|
import hypothesis
|
||||||
@@ -11,29 +14,31 @@ if PYTHON_VERSION > 2.6:
|
|||||||
|
|
||||||
# uncompyle6
|
# uncompyle6
|
||||||
|
|
||||||
|
|
||||||
@st.composite
|
@st.composite
|
||||||
def expressions(draw):
|
def expressions(draw):
|
||||||
# todo : would be nice to generate expressions using hypothesis however
|
# todo : would be nice to generate expressions using hypothesis however
|
||||||
# this is pretty involved so for now just use a corpus of expressions
|
# this is pretty involved so for now just use a corpus of expressions
|
||||||
# from which to select.
|
# from which to select.
|
||||||
return draw(st.sampled_from((
|
return draw(
|
||||||
'abc',
|
st.sampled_from(
|
||||||
'len(items)',
|
(
|
||||||
'x + 1',
|
"abc",
|
||||||
'lineno',
|
"len(items)",
|
||||||
'container',
|
"x + 1",
|
||||||
'self.attribute',
|
"lineno",
|
||||||
'self.method()',
|
"container",
|
||||||
# These expressions are failing, I think these are control
|
"self.attribute",
|
||||||
# flow problems rather than problems with FORMAT_VALUE,
|
"self.method()",
|
||||||
# however I need to confirm this...
|
# These expressions are failing, I think these are control
|
||||||
#'sorted(items, key=lambda x: x.name)',
|
# flow problems rather than problems with FORMAT_VALUE,
|
||||||
#'func(*args, **kwargs)',
|
# however I need to confirm this...
|
||||||
#'text or default',
|
#'sorted(items, key=lambda x: x.name)',
|
||||||
#'43 if life_the_universe and everything else None'
|
#'func(*args, **kwargs)',
|
||||||
)))
|
#'text or default',
|
||||||
|
#'43 if life_the_universe and everything else None'
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
@st.composite
|
@st.composite
|
||||||
def format_specifiers(draw):
|
def format_specifiers(draw):
|
||||||
@@ -54,36 +59,37 @@ if PYTHON_VERSION > 2.6:
|
|||||||
|
|
||||||
:return: An example format_specifier.
|
:return: An example format_specifier.
|
||||||
"""
|
"""
|
||||||
alphabet_strategy = st.characters(min_codepoint=ord('a'), max_codepoint=ord('z'))
|
alphabet_strategy = st.characters(
|
||||||
|
min_codepoint=ord("a"), max_codepoint=ord("z")
|
||||||
|
)
|
||||||
fill = draw(st.one_of(alphabet_strategy, st.none()))
|
fill = draw(st.one_of(alphabet_strategy, st.none()))
|
||||||
align = draw(st.sampled_from(list('<>=^')))
|
align = draw(st.sampled_from(list("<>=^")))
|
||||||
fill_align = (fill + align or '') if fill else ''
|
fill_align = (fill + align or "") if fill else ""
|
||||||
|
|
||||||
type_ = draw(st.sampled_from('bcdeEfFgGnosxX%'))
|
type_ = draw(st.sampled_from("bcdeEfFgGnosxX%"))
|
||||||
can_have_sign = type_ in 'deEfFgGnoxX%'
|
can_have_sign = type_ in "deEfFgGnoxX%"
|
||||||
can_have_comma = type_ in 'deEfFgG%'
|
can_have_comma = type_ in "deEfFgG%"
|
||||||
can_have_precision = type_ in 'fFgG'
|
can_have_precision = type_ in "fFgG"
|
||||||
can_have_pound = type_ in 'boxX%'
|
can_have_pound = type_ in "boxX%"
|
||||||
can_have_zero = type_ in 'oxX'
|
can_have_zero = type_ in "oxX"
|
||||||
|
|
||||||
sign = draw(st.sampled_from(list('+- ') + [''])) if can_have_sign else ''
|
sign = draw(st.sampled_from(list("+- ") + [""])) if can_have_sign else ""
|
||||||
pound = draw(st.sampled_from(('#', '',))) if can_have_pound else ''
|
pound = draw(st.sampled_from(("#", ""))) if can_have_pound else ""
|
||||||
zero = draw(st.sampled_from(('0', '',))) if can_have_zero else ''
|
zero = draw(st.sampled_from(("0", ""))) if can_have_zero else ""
|
||||||
|
|
||||||
int_strategy = st.integers(min_value=1, max_value=1000)
|
int_strategy = st.integers(min_value=1, max_value=1000)
|
||||||
|
|
||||||
width = draw(st.one_of(int_strategy, st.none()))
|
width = draw(st.one_of(int_strategy, st.none()))
|
||||||
width = str(width) if width is not None else ''
|
width = str(width) if width is not None else ""
|
||||||
|
|
||||||
comma = draw(st.sampled_from((',', '',))) if can_have_comma else ''
|
comma = draw(st.sampled_from((",", ""))) if can_have_comma else ""
|
||||||
if can_have_precision:
|
if can_have_precision:
|
||||||
precision = draw(st.one_of(int_strategy, st.none()))
|
precision = draw(st.one_of(int_strategy, st.none()))
|
||||||
precision = '.' + str(precision) if precision else ''
|
precision = "." + str(precision) if precision else ""
|
||||||
else:
|
else:
|
||||||
precision = ''
|
precision = ""
|
||||||
|
|
||||||
return ''.join((fill_align, sign, pound, zero, width, comma, precision, type_,))
|
|
||||||
|
|
||||||
|
return "".join((fill_align, sign, pound, zero, width, comma, precision, type_))
|
||||||
|
|
||||||
@st.composite
|
@st.composite
|
||||||
def fstrings(draw):
|
def fstrings(draw):
|
||||||
@@ -96,9 +102,7 @@ if PYTHON_VERSION > 2.6:
|
|||||||
:return: A valid f-string.
|
:return: A valid f-string.
|
||||||
"""
|
"""
|
||||||
character_strategy = st.characters(
|
character_strategy = st.characters(
|
||||||
blacklist_characters='\r\n\'\\s{}',
|
blacklist_characters="\r\n'\\s{}", min_codepoint=1, max_codepoint=1000
|
||||||
min_codepoint=1,
|
|
||||||
max_codepoint=1000,
|
|
||||||
)
|
)
|
||||||
is_raw = draw(st.booleans())
|
is_raw = draw(st.booleans())
|
||||||
integer_strategy = st.integers(min_value=0, max_value=3)
|
integer_strategy = st.integers(min_value=0, max_value=3)
|
||||||
@@ -106,49 +110,49 @@ if PYTHON_VERSION > 2.6:
|
|||||||
content = []
|
content = []
|
||||||
for _ in range(expression_count):
|
for _ in range(expression_count):
|
||||||
expression = draw(expressions())
|
expression = draw(expressions())
|
||||||
conversion = draw(st.sampled_from(('', '!s', '!r', '!a',)))
|
conversion = draw(st.sampled_from(("", "!s", "!r", "!a")))
|
||||||
has_specifier = draw(st.booleans())
|
has_specifier = draw(st.booleans())
|
||||||
specifier = ':' + draw(format_specifiers()) if has_specifier else ''
|
specifier = ":" + draw(format_specifiers()) if has_specifier else ""
|
||||||
content.append('{{{}{}}}'.format(expression, conversion, specifier))
|
content.append("{{{}{}}}".format(expression, conversion, specifier))
|
||||||
content.append(draw(st.text(character_strategy)))
|
content.append(draw(st.text(character_strategy)))
|
||||||
content = ''.join(content)
|
content = "".join(content)
|
||||||
return "f{}'{}'".format('r' if is_raw else '', content)
|
return "f{}'{}'".format("r" if is_raw else "", content)
|
||||||
|
|
||||||
|
@pytest.mark.skipif(PYTHON_VERSION != 3.6, reason="need Python 3.6")
|
||||||
@pytest.mark.skipif(PYTHON_VERSION != 3.6, reason='need Python 3.6')
|
|
||||||
@hypothesis.given(format_specifiers())
|
@hypothesis.given(format_specifiers())
|
||||||
def test_format_specifiers(format_specifier):
|
def test_format_specifiers(format_specifier):
|
||||||
"""Verify that format_specifiers generates valid specifiers"""
|
"""Verify that format_specifiers generates valid specifiers"""
|
||||||
try:
|
try:
|
||||||
exec('"{:' + format_specifier + '}".format(0)')
|
exec('"{:' + format_specifier + '}".format(0)')
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
if 'Unknown format code' not in str(e):
|
if "Unknown format code" not in str(e):
|
||||||
raise
|
raise
|
||||||
|
|
||||||
|
|
||||||
def run_test(text):
|
def run_test(text):
|
||||||
hypothesis.assume(len(text))
|
hypothesis.assume(len(text))
|
||||||
hypothesis.assume("f'{" in text)
|
hypothesis.assume("f'{" in text)
|
||||||
expr = text + '\n'
|
expr = text + "\n"
|
||||||
code = compile(expr, '<string>', 'single')
|
code = compile(expr, "<string>", "single")
|
||||||
deparsed = deparse_code(PYTHON_VERSION, code, compile_mode='single')
|
deparsed = code_deparse(code, sys.stdout, PYTHON_VERSION, compile_mode="single")
|
||||||
recompiled = compile(deparsed.text, '<string>', 'single')
|
recompiled = compile(deparsed.text, "<string>", "single")
|
||||||
if recompiled != code:
|
if recompiled != code:
|
||||||
assert 'dis(' + deparsed.text.strip('\n') + ')' == 'dis(' + expr.strip('\n') + ')'
|
print(recompiled)
|
||||||
|
print("================")
|
||||||
|
print(code)
|
||||||
|
print("----------------")
|
||||||
|
assert (
|
||||||
|
"dis(" + deparsed.text.strip("\n") + ")"
|
||||||
|
== "dis(" + expr.strip("\n") + ")"
|
||||||
|
)
|
||||||
|
|
||||||
|
@pytest.mark.skipif(PYTHON_VERSION != 3.6, reason="need Python 3.6")
|
||||||
@pytest.mark.skipif(PYTHON_VERSION != 3.6, reason='need Python 3.6')
|
|
||||||
@hypothesis.given(fstrings())
|
@hypothesis.given(fstrings())
|
||||||
def test_uncompyle_fstring(fstring):
|
def test_uncompyle_fstring(fstring):
|
||||||
"""Verify uncompyling fstring bytecode"""
|
"""Verify uncompyling fstring bytecode"""
|
||||||
run_test(fstring)
|
run_test(fstring)
|
||||||
|
|
||||||
|
@pytest.mark.skipif(PYTHON_VERSION != 3.6, reason="need Python 3.6+")
|
||||||
@pytest.mark.skipif(PYTHON_VERSION < 3.6, reason='need Python 3.6+')
|
@pytest.mark.parametrize("fstring", ["f'{abc}{abc!s}'", "f'{abc}0'"])
|
||||||
@pytest.mark.parametrize('fstring', [
|
|
||||||
"f'{abc}{abc!s}'",
|
|
||||||
"f'{abc}0'",
|
|
||||||
])
|
|
||||||
def test_uncompyle_direct(fstring):
|
def test_uncompyle_direct(fstring):
|
||||||
"""useful for debugging"""
|
"""useful for debugging"""
|
||||||
run_test(fstring)
|
run_test(fstring)
|
||||||
|
@@ -1,72 +1,100 @@
|
|||||||
import re
|
import re
|
||||||
from uncompyle6 import PYTHON_VERSION, PYTHON3, IS_PYPY # , PYTHON_VERSION
|
from uncompyle6 import PYTHON_VERSION, PYTHON3, IS_PYPY # , PYTHON_VERSION
|
||||||
from uncompyle6.parser import get_python_parser, python_parser
|
from uncompyle6.parser import get_python_parser, python_parser
|
||||||
from uncompyle6.scanner import get_scanner
|
from uncompyle6.scanner import get_scanner
|
||||||
|
|
||||||
def test_grammar():
|
|
||||||
|
|
||||||
|
def test_grammar():
|
||||||
def check_tokens(tokens, opcode_set):
|
def check_tokens(tokens, opcode_set):
|
||||||
remain_tokens = set(tokens) - opcode_set
|
remain_tokens = set(tokens) - opcode_set
|
||||||
remain_tokens = set([re.sub(r'_\d+$','', t) for t in remain_tokens])
|
remain_tokens = set([re.sub(r"_\d+$", "", t) for t in remain_tokens])
|
||||||
remain_tokens = set([re.sub('_CONT$','', t) for t in remain_tokens])
|
remain_tokens = set([re.sub("_CONT$", "", t) for t in remain_tokens])
|
||||||
|
remain_tokens = set([re.sub("LOAD_CODE$", "", t) for t in remain_tokens])
|
||||||
remain_tokens = set(remain_tokens) - opcode_set
|
remain_tokens = set(remain_tokens) - opcode_set
|
||||||
assert remain_tokens == set([]), \
|
assert remain_tokens == set([]), "Remaining tokens %s\n====\n%s" % (
|
||||||
"Remaining tokens %s\n====\n%s" % (remain_tokens, p.dump_grammar())
|
remain_tokens,
|
||||||
|
p.dump_grammar(),
|
||||||
|
)
|
||||||
|
|
||||||
p = get_python_parser(PYTHON_VERSION, is_pypy=IS_PYPY)
|
p = get_python_parser(PYTHON_VERSION, is_pypy=IS_PYPY)
|
||||||
(lhs, rhs, tokens,
|
(lhs, rhs, tokens, right_recursive, dup_rhs) = p.check_sets()
|
||||||
right_recursive, dup_rhs) = p.check_sets()
|
|
||||||
|
|
||||||
# We have custom rules that create the below
|
# We have custom rules that create the below
|
||||||
expect_lhs = set(['pos_arg', 'get_iter', 'attribute'])
|
expect_lhs = set(["pos_arg"])
|
||||||
|
|
||||||
unused_rhs = set(['list', 'mkfunc',
|
if PYTHON_VERSION < 3.8:
|
||||||
'mklambda',
|
if PYTHON_VERSION < 3.7:
|
||||||
'unpack',])
|
expect_lhs.add("attribute")
|
||||||
expect_right_recursive = set([('designList',
|
|
||||||
('store', 'DUP_TOP', 'designList'))])
|
|
||||||
|
|
||||||
if PYTHON_VERSION != 3.7:
|
expect_lhs.add("get_iter")
|
||||||
unused_rhs.add('call')
|
else:
|
||||||
|
expect_lhs.add("async_with_as_stmt")
|
||||||
|
expect_lhs.add("async_with_stmt")
|
||||||
|
|
||||||
|
unused_rhs = set(["list", "mkfunc", "mklambda", "unpack"])
|
||||||
|
|
||||||
|
expect_right_recursive = set([("designList", ("store", "DUP_TOP", "designList"))])
|
||||||
|
|
||||||
|
if PYTHON_VERSION <= 3.7:
|
||||||
|
unused_rhs.add("call")
|
||||||
|
|
||||||
if PYTHON_VERSION > 2.6:
|
if PYTHON_VERSION > 2.6:
|
||||||
expect_lhs.add('kvlist')
|
expect_lhs.add("kvlist")
|
||||||
expect_lhs.add('kv3')
|
expect_lhs.add("kv3")
|
||||||
unused_rhs.add('dict')
|
unused_rhs.add("dict")
|
||||||
|
|
||||||
if PYTHON3:
|
if PYTHON3:
|
||||||
expect_lhs.add('load_genexpr')
|
expect_lhs.add("load_genexpr")
|
||||||
|
|
||||||
unused_rhs = unused_rhs.union(set("""
|
unused_rhs = unused_rhs.union(
|
||||||
|
set(
|
||||||
|
"""
|
||||||
except_pop_except generator_exp
|
except_pop_except generator_exp
|
||||||
""".split()))
|
""".split()
|
||||||
|
)
|
||||||
|
)
|
||||||
if PYTHON_VERSION >= 3.0:
|
if PYTHON_VERSION >= 3.0:
|
||||||
expect_lhs.add("annotate_arg")
|
if PYTHON_VERSION < 3.7:
|
||||||
expect_lhs.add("annotate_tuple")
|
expect_lhs.add("annotate_arg")
|
||||||
unused_rhs.add("mkfunc_annotate")
|
expect_lhs.add("annotate_tuple")
|
||||||
|
unused_rhs.add("mkfunc_annotate")
|
||||||
|
|
||||||
unused_rhs.add("dict_comp")
|
unused_rhs.add("dict_comp")
|
||||||
unused_rhs.add("classdefdeco1")
|
unused_rhs.add("classdefdeco1")
|
||||||
unused_rhs.add("tryelsestmtl")
|
unused_rhs.add("tryelsestmtl")
|
||||||
if PYTHON_VERSION >= 3.5:
|
if PYTHON_VERSION >= 3.5:
|
||||||
expect_right_recursive.add((('l_stmts',
|
expect_right_recursive.add(
|
||||||
('lastl_stmt', 'come_froms', 'l_stmts'))))
|
(("l_stmts", ("lastl_stmt", "come_froms", "l_stmts")))
|
||||||
|
)
|
||||||
pass
|
pass
|
||||||
elif 3.0 < PYTHON_VERSION < 3.3:
|
elif 3.0 < PYTHON_VERSION < 3.3:
|
||||||
expect_right_recursive.add((('l_stmts',
|
expect_right_recursive.add(
|
||||||
('lastl_stmt', 'COME_FROM', 'l_stmts'))))
|
(("l_stmts", ("lastl_stmt", "COME_FROM", "l_stmts")))
|
||||||
|
)
|
||||||
pass
|
pass
|
||||||
pass
|
pass
|
||||||
pass
|
pass
|
||||||
else:
|
else:
|
||||||
expect_lhs.add('kwarg')
|
expect_lhs.add("kwarg")
|
||||||
|
|
||||||
assert expect_lhs == set(lhs)
|
assert expect_lhs == set(lhs)
|
||||||
assert unused_rhs == set(rhs)
|
|
||||||
|
# FIXME
|
||||||
|
if PYTHON_VERSION != 3.8:
|
||||||
|
assert unused_rhs == set(rhs)
|
||||||
|
|
||||||
assert expect_right_recursive == right_recursive
|
assert expect_right_recursive == right_recursive
|
||||||
|
|
||||||
expect_dup_rhs = frozenset([('COME_FROM',), ('CONTINUE',), ('JUMP_ABSOLUTE',),
|
expect_dup_rhs = frozenset(
|
||||||
('LOAD_CONST',),
|
[
|
||||||
('JUMP_BACK',), ('JUMP_FORWARD',)])
|
("COME_FROM",),
|
||||||
|
("CONTINUE",),
|
||||||
|
("JUMP_ABSOLUTE",),
|
||||||
|
("LOAD_CONST",),
|
||||||
|
("JUMP_BACK",),
|
||||||
|
("JUMP_FORWARD",),
|
||||||
|
]
|
||||||
|
)
|
||||||
reduced_dup_rhs = dict((k, dup_rhs[k]) for k in dup_rhs if k not in expect_dup_rhs)
|
reduced_dup_rhs = dict((k, dup_rhs[k]) for k in dup_rhs if k not in expect_dup_rhs)
|
||||||
for k in reduced_dup_rhs:
|
for k in reduced_dup_rhs:
|
||||||
print(k, reduced_dup_rhs[k])
|
print(k, reduced_dup_rhs[k])
|
||||||
@@ -74,31 +102,42 @@ def test_grammar():
|
|||||||
|
|
||||||
s = get_scanner(PYTHON_VERSION, IS_PYPY)
|
s = get_scanner(PYTHON_VERSION, IS_PYPY)
|
||||||
ignore_set = set(
|
ignore_set = set(
|
||||||
"""
|
"""
|
||||||
JUMP_BACK CONTINUE
|
JUMP_BACK CONTINUE
|
||||||
COME_FROM COME_FROM_EXCEPT
|
COME_FROM COME_FROM_EXCEPT
|
||||||
COME_FROM_EXCEPT_CLAUSE
|
COME_FROM_EXCEPT_CLAUSE
|
||||||
COME_FROM_LOOP COME_FROM_WITH
|
COME_FROM_LOOP COME_FROM_WITH
|
||||||
COME_FROM_FINALLY ELSE
|
COME_FROM_FINALLY ELSE
|
||||||
LOAD_GENEXPR LOAD_ASSERT LOAD_SETCOMP LOAD_DICTCOMP
|
LOAD_GENEXPR LOAD_ASSERT LOAD_SETCOMP LOAD_DICTCOMP LOAD_STR LOAD_CODE
|
||||||
LAMBDA_MARKER
|
LAMBDA_MARKER
|
||||||
RETURN_END_IF RETURN_END_IF_LAMBDA RETURN_VALUE_LAMBDA RETURN_LAST
|
RETURN_END_IF RETURN_END_IF_LAMBDA RETURN_VALUE_LAMBDA RETURN_LAST
|
||||||
""".split())
|
""".split()
|
||||||
|
)
|
||||||
if 2.6 <= PYTHON_VERSION <= 2.7:
|
if 2.6 <= PYTHON_VERSION <= 2.7:
|
||||||
opcode_set = set(s.opc.opname).union(ignore_set)
|
opcode_set = set(s.opc.opname).union(ignore_set)
|
||||||
if PYTHON_VERSION == 2.6:
|
if PYTHON_VERSION == 2.6:
|
||||||
opcode_set.add("THEN")
|
opcode_set.add("THEN")
|
||||||
check_tokens(tokens, opcode_set)
|
check_tokens(tokens, opcode_set)
|
||||||
elif PYTHON_VERSION == 3.4:
|
elif PYTHON_VERSION == 3.4:
|
||||||
ignore_set.add('LOAD_CLASSNAME')
|
ignore_set.add("LOAD_CLASSNAME")
|
||||||
ignore_set.add('STORE_LOCALS')
|
ignore_set.add("STORE_LOCALS")
|
||||||
opcode_set = set(s.opc.opname).union(ignore_set)
|
opcode_set = set(s.opc.opname).union(ignore_set)
|
||||||
check_tokens(tokens, opcode_set)
|
check_tokens(tokens, opcode_set)
|
||||||
|
|
||||||
|
|
||||||
def test_dup_rule():
|
def test_dup_rule():
|
||||||
import inspect
|
import inspect
|
||||||
python_parser(PYTHON_VERSION, inspect.currentframe().f_code,
|
|
||||||
is_pypy=IS_PYPY,
|
python_parser(
|
||||||
parser_debug={
|
PYTHON_VERSION,
|
||||||
'dups': True, 'transition': False, 'reduce': False,
|
inspect.currentframe().f_code,
|
||||||
'rules': False, 'errorstack': None, 'context': True})
|
is_pypy=IS_PYPY,
|
||||||
|
parser_debug={
|
||||||
|
"dups": True,
|
||||||
|
"transition": False,
|
||||||
|
"reduce": False,
|
||||||
|
"rules": False,
|
||||||
|
"errorstack": None,
|
||||||
|
"context": True,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
@@ -15,7 +15,7 @@ else:
|
|||||||
def iteritems(d):
|
def iteritems(d):
|
||||||
return d.iteritems()
|
return d.iteritems()
|
||||||
|
|
||||||
from uncompyle6.semantics.pysource import SourceWalker as SourceWalker
|
from uncompyle6.semantics.pysource import (SourceWalker, deparse_code2str)
|
||||||
|
|
||||||
def test_template_engine():
|
def test_template_engine():
|
||||||
s = StringIO()
|
s = StringIO()
|
||||||
@@ -127,11 +127,17 @@ def test_tables():
|
|||||||
"Full entry: %s" %
|
"Full entry: %s" %
|
||||||
(name, k, arg, typ, entry[arg], type(entry[arg]), entry)
|
(name, k, arg, typ, entry[arg], type(entry[arg]), entry)
|
||||||
)
|
)
|
||||||
assert len(tup) == 2
|
assert 2 <= len(tup) <= 3
|
||||||
for j, x in enumerate(tup):
|
for j, x in enumerate(tup):
|
||||||
assert isinstance(x, int), (
|
if len(tup) == 3 and j == 1:
|
||||||
"%s[%s][%d][%d] type '%s' is '%s should be an int but is %s. Full entry: %s" %
|
assert isinstance(x, str), (
|
||||||
(name, k, arg, j, typ, x, type(x), entry)
|
"%s[%s][%d][%d] type '%s' is '%s should be an string but is %s. Full entry: %s" %
|
||||||
|
(name, k, arg, j, typ, x, type(x), entry)
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
assert isinstance(x, int), (
|
||||||
|
"%s[%s][%d][%d] type '%s' is '%s should be an int but is %s. Full entry: %s" %
|
||||||
|
(name, k, arg, j, typ, x, type(x), entry)
|
||||||
)
|
)
|
||||||
pass
|
pass
|
||||||
arg += 1
|
arg += 1
|
||||||
@@ -179,3 +185,11 @@ def test_tables():
|
|||||||
assert arg == len(entry), (
|
assert arg == len(entry), (
|
||||||
"%s[%s] arg %d should be length of entry %d. Full entry: %s" %
|
"%s[%s] arg %d should be length of entry %d. Full entry: %s" %
|
||||||
(name, k, arg, len(entry), entry))
|
(name, k, arg, len(entry), entry))
|
||||||
|
|
||||||
|
def test_deparse_code2str():
|
||||||
|
def deparse_test(co):
|
||||||
|
"This is a docstring"
|
||||||
|
s = deparse_code2str(co, debug_opts={"asm": "after", "tree": True})
|
||||||
|
assert s
|
||||||
|
return
|
||||||
|
deparse_test(deparse_test.__code__)
|
||||||
|
@@ -1,3 +1,4 @@
|
|||||||
|
from uncompyle6 import PYTHON_VERSION
|
||||||
from uncompyle6.scanners.tok import Token
|
from uncompyle6.scanners.tok import Token
|
||||||
|
|
||||||
def test_token():
|
def test_token():
|
||||||
@@ -16,7 +17,7 @@ def test_token():
|
|||||||
# Make sure formatting of: LOAD_CONST False. We assume False is the 0th index
|
# Make sure formatting of: LOAD_CONST False. We assume False is the 0th index
|
||||||
# of co_consts.
|
# of co_consts.
|
||||||
t = Token('LOAD_CONST', offset=1, attr=False, pattr=False, has_arg=True)
|
t = Token('LOAD_CONST', offset=1, attr=False, pattr=False, has_arg=True)
|
||||||
expect = ' 1 LOAD_CONST 0 False'
|
expect = ' 1 LOAD_CONST False'
|
||||||
assert t.format() == expect
|
assert t.format() == expect
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
2
pytest/testdata/if-2.7.right
vendored
2
pytest/testdata/if-2.7.right
vendored
@@ -8,5 +8,5 @@
|
|||||||
9 STORE_NAME 2 'b'
|
9 STORE_NAME 2 'b'
|
||||||
12 JUMP_FORWARD 0 'to 15'
|
12 JUMP_FORWARD 0 'to 15'
|
||||||
15_0 COME_FROM 12 '12'
|
15_0 COME_FROM 12 '12'
|
||||||
15 LOAD_CONST 0 None
|
15 LOAD_CONST None
|
||||||
18 RETURN_VALUE
|
18 RETURN_VALUE
|
||||||
|
6
pytest/testdata/ifelse-2.7.right
vendored
6
pytest/testdata/ifelse-2.7.right
vendored
@@ -4,12 +4,12 @@
|
|||||||
3 0 LOAD_NAME 0 'True'
|
3 0 LOAD_NAME 0 'True'
|
||||||
3 POP_JUMP_IF_FALSE 15 'to 15'
|
3 POP_JUMP_IF_FALSE 15 'to 15'
|
||||||
|
|
||||||
4 6 LOAD_CONST 0 1
|
4 6 LOAD_CONST 1
|
||||||
9 STORE_NAME 1 'b'
|
9 STORE_NAME 1 'b'
|
||||||
12 JUMP_FORWARD 6 'to 21'
|
12 JUMP_FORWARD 6 'to 21'
|
||||||
|
|
||||||
6 15 LOAD_CONST 1 2
|
6 15 LOAD_CONST 2
|
||||||
18 STORE_NAME 2 'd'
|
18 STORE_NAME 2 'd'
|
||||||
21_0 COME_FROM 12 '12'
|
21_0 COME_FROM 12 '12'
|
||||||
21 LOAD_CONST 2 None
|
21 LOAD_CONST None
|
||||||
24 RETURN_VALUE
|
24 RETURN_VALUE
|
||||||
|
@@ -1,16 +1,20 @@
|
|||||||
# future
|
# future
|
||||||
from __future__ import print_function
|
from __future__ import print_function
|
||||||
|
|
||||||
# std
|
# std
|
||||||
import os
|
import os
|
||||||
import difflib
|
import difflib
|
||||||
import subprocess
|
import subprocess
|
||||||
import tempfile
|
import tempfile
|
||||||
import functools
|
import functools
|
||||||
|
|
||||||
# uncompyle6 / xdis
|
# uncompyle6 / xdis
|
||||||
from uncompyle6 import PYTHON_VERSION, PYTHON3, IS_PYPY, deparse_code
|
from uncompyle6 import PYTHON_VERSION, PYTHON3, IS_PYPY, code_deparse
|
||||||
|
|
||||||
# TODO : I think we can get xdis to support the dis api (python 3 version) by doing something like this there
|
# TODO : I think we can get xdis to support the dis api (python 3 version) by doing something like this there
|
||||||
from xdis.bytecode import Bytecode
|
from xdis.bytecode import Bytecode
|
||||||
from xdis.main import get_opcode
|
from xdis.main import get_opcode
|
||||||
|
|
||||||
opc = get_opcode(PYTHON_VERSION, IS_PYPY)
|
opc = get_opcode(PYTHON_VERSION, IS_PYPY)
|
||||||
Bytecode = functools.partial(Bytecode, opc=opc)
|
Bytecode = functools.partial(Bytecode, opc=opc)
|
||||||
import six
|
import six
|
||||||
@@ -20,6 +24,7 @@ if PYTHON3:
|
|||||||
else:
|
else:
|
||||||
from StringIO import StringIO
|
from StringIO import StringIO
|
||||||
|
|
||||||
|
|
||||||
def _dis_to_text(co):
|
def _dis_to_text(co):
|
||||||
return Bytecode(co).dis()
|
return Bytecode(co).dis()
|
||||||
|
|
||||||
@@ -33,36 +38,32 @@ def print_diff(original, uncompyled):
|
|||||||
:param original: Text describing the original code object.
|
:param original: Text describing the original code object.
|
||||||
:param uncompyled: Text describing the uncompyled code object.
|
:param uncompyled: Text describing the uncompyled code object.
|
||||||
"""
|
"""
|
||||||
original_lines = original.split('\n')
|
original_lines = original.split("\n")
|
||||||
uncompyled_lines = uncompyled.split('\n')
|
uncompyled_lines = uncompyled.split("\n")
|
||||||
args = original_lines, uncompyled_lines, 'original', 'uncompyled'
|
args = original_lines, uncompyled_lines, "original", "uncompyled"
|
||||||
try:
|
try:
|
||||||
from bs4 import BeautifulSoup
|
from bs4 import BeautifulSoup
|
||||||
|
|
||||||
diff = difflib.HtmlDiff().make_file(*args)
|
diff = difflib.HtmlDiff().make_file(*args)
|
||||||
diff = BeautifulSoup(diff, "html.parser")
|
diff = BeautifulSoup(diff, "html.parser")
|
||||||
diff.select_one('table[summary="Legends"]').extract()
|
diff.select_one('table[summary="Legends"]').extract()
|
||||||
except ImportError:
|
except ImportError:
|
||||||
print('\nTo display diff highlighting run:\n pip install BeautifulSoup4')
|
print("\nTo display diff highlighting run:\n pip install BeautifulSoup4")
|
||||||
diff = difflib.HtmlDiff().make_table(*args)
|
diff = difflib.HtmlDiff().make_table(*args)
|
||||||
|
|
||||||
with tempfile.NamedTemporaryFile(delete=False) as f:
|
with tempfile.NamedTemporaryFile(delete=False) as f:
|
||||||
f.write(str(diff).encode('utf-8'))
|
f.write(str(diff).encode("utf-8"))
|
||||||
|
|
||||||
try:
|
try:
|
||||||
print()
|
print()
|
||||||
html = subprocess.check_output([
|
html = subprocess.check_output(
|
||||||
'elinks',
|
["elinks", "-dump", "-no-references", "-dump-color-mode", "1", f.name]
|
||||||
'-dump',
|
).decode("utf-8")
|
||||||
'-no-references',
|
|
||||||
'-dump-color-mode',
|
|
||||||
'1',
|
|
||||||
f.name,
|
|
||||||
]).decode('utf-8')
|
|
||||||
print(html)
|
print(html)
|
||||||
except:
|
except:
|
||||||
print('\nFor side by side diff install elinks')
|
print("\nFor side by side diff install elinks")
|
||||||
diff = difflib.Differ().compare(original_lines, uncompyled_lines)
|
diff = difflib.Differ().compare(original_lines, uncompyled_lines)
|
||||||
print('\n'.join(diff))
|
print("\n".join(diff))
|
||||||
finally:
|
finally:
|
||||||
os.unlink(f.name)
|
os.unlink(f.name)
|
||||||
|
|
||||||
@@ -80,18 +81,19 @@ def are_instructions_equal(i1, i2):
|
|||||||
|
|
||||||
:return: True if the two instructions are approximately equal, otherwise False.
|
:return: True if the two instructions are approximately equal, otherwise False.
|
||||||
"""
|
"""
|
||||||
result = (1==1
|
result = (
|
||||||
|
1 == 1
|
||||||
and i1.opname == i2.opname
|
and i1.opname == i2.opname
|
||||||
and i1.opcode == i2.opcode
|
and i1.opcode == i2.opcode
|
||||||
and i1.arg == i2.arg
|
and i1.arg == i2.arg
|
||||||
# ignore differences due to code objects
|
# ignore differences due to code objects
|
||||||
# TODO : Better way of ignoring address
|
# TODO : Better way of ignoring address
|
||||||
and (i1.argval == i2.argval or '<code object' in str(i1.argval))
|
and (i1.argval == i2.argval or "<code object" in str(i1.argval))
|
||||||
# TODO : Should probably recurse to check code objects
|
# TODO : Should probably recurse to check code objects
|
||||||
and (i1.argrepr == i2.argrepr or '<code object' in i1.argrepr)
|
and (i1.argrepr == i2.argrepr or "<code object" in i1.argrepr)
|
||||||
and i1.offset == i2.offset
|
and i1.offset == i2.offset
|
||||||
# ignore differences in line numbers
|
# ignore differences in line numbers
|
||||||
#and i1.starts_line
|
# and i1.starts_line
|
||||||
and i1.is_jump_target == i2.is_jump_target
|
and i1.is_jump_target == i2.is_jump_target
|
||||||
)
|
)
|
||||||
return result
|
return result
|
||||||
@@ -115,22 +117,21 @@ def are_code_objects_equal(co1, co2):
|
|||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
def validate_uncompyle(text, mode='exec'):
|
def validate_uncompyle(text, mode="exec"):
|
||||||
"""
|
"""
|
||||||
Validate decompilation of the given source code.
|
Validate decompilation of the given source code.
|
||||||
|
|
||||||
:param text: Source to validate decompilation of.
|
:param text: Source to validate decompilation of.
|
||||||
"""
|
"""
|
||||||
original_code = compile(text, '<string>', mode)
|
original_code = compile(text, "<string>", mode)
|
||||||
original_dis = _dis_to_text(original_code)
|
original_dis = _dis_to_text(original_code)
|
||||||
original_text = text
|
original_text = text
|
||||||
|
|
||||||
deparsed = deparse_code(PYTHON_VERSION, original_code,
|
deparsed = code_deparse(
|
||||||
compile_mode=mode,
|
original_code, out=six.StringIO(), version=PYTHON_VERSION, compile_mode=mode
|
||||||
out=six.StringIO(),
|
)
|
||||||
is_pypy=IS_PYPY)
|
|
||||||
uncompyled_text = deparsed.text
|
uncompyled_text = deparsed.text
|
||||||
uncompyled_code = compile(uncompyled_text, '<string>', 'exec')
|
uncompyled_code = compile(uncompyled_text, "<string>", "exec")
|
||||||
|
|
||||||
if not are_code_objects_equal(uncompyled_code, original_code):
|
if not are_code_objects_equal(uncompyled_code, original_code):
|
||||||
|
|
||||||
@@ -138,15 +139,17 @@ def validate_uncompyle(text, mode='exec'):
|
|||||||
|
|
||||||
def output(text, dis):
|
def output(text, dis):
|
||||||
width = 60
|
width = 60
|
||||||
return '\n\n'.join([
|
return "\n\n".join(
|
||||||
' SOURCE CODE '.center(width, '#'),
|
[
|
||||||
text.strip(),
|
" SOURCE CODE ".center(width, "#"),
|
||||||
' BYTECODE '.center(width, '#'),
|
text.strip(),
|
||||||
dis
|
" BYTECODE ".center(width, "#"),
|
||||||
])
|
dis,
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
original = output(original_text, original_dis)
|
original = output(original_text, original_dis)
|
||||||
uncompyled = output(uncompyled_text, uncompyled_dis)
|
uncompyled = output(uncompyled_text, uncompyled_dis)
|
||||||
print_diff(original, uncompyled)
|
print_diff(original, uncompyled)
|
||||||
|
|
||||||
assert 'original' == 'uncompyled'
|
assert "original" == "uncompyled"
|
||||||
|
@@ -1,2 +1,4 @@
|
|||||||
flake8
|
flake8
|
||||||
hypothesis<=3.0.0
|
hypothesis<=3.0.0
|
||||||
|
six
|
||||||
|
pytest==3.2.5
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
[bdist_rpm]
|
[bdist_rpm]
|
||||||
release = 1
|
release = 1
|
||||||
packager = Mysterie <kajusska@gmail.com>
|
packager = rocky <rb@dustyfeet.com>
|
||||||
doc_files = README
|
doc_files = README
|
||||||
# CHANGES.txt
|
# CHANGES.txt
|
||||||
# USAGE.txt
|
# USAGE.txt
|
||||||
@@ -8,4 +8,4 @@ doc_files = README
|
|||||||
# examples/
|
# examples/
|
||||||
|
|
||||||
[bdist_wheel]
|
[bdist_wheel]
|
||||||
universal=1
|
# universal=1
|
||||||
|
5
setup.py
5
setup.py
@@ -4,8 +4,8 @@ import sys
|
|||||||
"""Setup script for the 'uncompyle6' distribution."""
|
"""Setup script for the 'uncompyle6' distribution."""
|
||||||
|
|
||||||
SYS_VERSION = sys.version_info[0:2]
|
SYS_VERSION = sys.version_info[0:2]
|
||||||
if not ((2, 6) <= SYS_VERSION <= (3, 7)):
|
if not ((2, 6) <= SYS_VERSION <= (3, 9)):
|
||||||
mess = "Python Release 2.6 .. 3.7 are supported in this code branch."
|
mess = "Python Release 2.6 .. 3.9 are supported in this code branch."
|
||||||
if ((2, 4) <= SYS_VERSION <= (2, 7)):
|
if ((2, 4) <= SYS_VERSION <= (2, 7)):
|
||||||
mess += ("\nFor your Python, version %s, use the python-2.4 code/branch." %
|
mess += ("\nFor your Python, version %s, use the python-2.4 code/branch." %
|
||||||
sys.version[0:3])
|
sys.version[0:3])
|
||||||
@@ -32,6 +32,7 @@ setup(
|
|||||||
install_requires = install_requires,
|
install_requires = install_requires,
|
||||||
license = license,
|
license = license,
|
||||||
long_description = long_description,
|
long_description = long_description,
|
||||||
|
long_description_content_type = "text/x-rst",
|
||||||
name = modname,
|
name = modname,
|
||||||
packages = find_packages(),
|
packages = find_packages(),
|
||||||
py_modules = py_modules,
|
py_modules = py_modules,
|
||||||
|
107
test/Makefile
107
test/Makefile
@@ -1,5 +1,6 @@
|
|||||||
PHONY=check clean dist distclean test test-unit test-functional rmChangeLog clean_pyc nosetests \
|
PHONY=check clean dist distclean test test-unit test-functional rmChangeLog clean_pyc nosetests \
|
||||||
check-bytecode-1 check-bytecode-1.3 check-bytecode-1.4 check-bytecode-1.5 \
|
check-bytecode-1.0 check-bytecode-1.1 check-bytecode-1.2 check-bytecode-1.3 \
|
||||||
|
check-bytecode-1 check-bytecode-1.4 check-bytecode-1.5 check-bytecode-1.6 \
|
||||||
check-bytecode-2 check-bytecode-3 check-bytecode-3-short \
|
check-bytecode-2 check-bytecode-3 check-bytecode-3-short \
|
||||||
check-bytecode-2.2 check-byteocde-2.3 check-bytecode-2.4 \
|
check-bytecode-2.2 check-byteocde-2.3 check-bytecode-2.4 \
|
||||||
check-short check-2.6 check-2.7 check-3.0 check-3.1 check-3.2 check-3.3 \
|
check-short check-2.6 check-2.7 check-3.0 check-3.1 check-3.2 check-3.3 \
|
||||||
@@ -33,43 +34,48 @@ check-2.6 check-2.7: check-bytecode-2 check-bytecode-3 check-bytecode-1 check-na
|
|||||||
|
|
||||||
#: Run working tests from Python 3.0
|
#: Run working tests from Python 3.0
|
||||||
check-3.0: check-bytecode
|
check-3.0: check-bytecode
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.0 --weak-verify $(COMPILE)
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.0-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.0-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.0 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
#: Run working tests from Python 3.1
|
#: Run working tests from Python 3.1
|
||||||
check-3.1: check-bytecode
|
check-3.1: check-bytecode
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.1 --weak-verify $(COMPILE)
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.1-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.1-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.1 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
#: Run working tests from Python 3.2
|
#: Run working tests from Python 3.2
|
||||||
check-3.2: check-bytecode
|
check-3.2: check-bytecode
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.2 --weak-verify $(COMPILE)
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.2-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.2-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.2 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
#: Run working tests from Python 3.3
|
#: Run working tests from Python 3.3
|
||||||
check-3.3: check-bytecode
|
check-3.3: check-bytecode
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.3 --weak-verify $(COMPILE)
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.3-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.3-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.3 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
#: Run working tests from Python 3.4
|
#: Run working tests from Python 3.4
|
||||||
check-3.4: check-bytecode check-3.4-ok check-2.7-ok
|
check-3.4: check-bytecode check-3.4-ok check-2.7-ok
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.4 --weak-verify $(COMPILE)
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.4-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.4-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.4 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
#: Run working tests from Python 3.5
|
#: Run working tests from Python 3.5
|
||||||
check-3.5: check-bytecode
|
check-3.5: check-bytecode
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.5 --weak-verify $(COMPILE)
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.5-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.5-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.5 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
#: Run working tests from Python 3.6
|
#: Run working tests from Python 3.6
|
||||||
check-3.6: check-bytecode
|
check-3.6: check-bytecode
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.6 --weak-verify $(COMPILE)
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.6-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.6-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.6 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
#: Run working tests from Python 3.7
|
#: Run working tests from Python 3.7
|
||||||
check-3.7: check-bytecode
|
check-3.7: check-bytecode
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.7 --weak-verify $(COMPILE)
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.7-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.7-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.7 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
|
#: Run working tests from Python 3.8
|
||||||
|
check-3.8: check-bytecode
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.8-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.8 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
# FIXME
|
# FIXME
|
||||||
#: this is called when running under pypy3.5-5.8.0 or pypy2-5.6.0
|
#: this is called when running under pypy3.5-5.8.0 or pypy2-5.6.0
|
||||||
@@ -80,7 +86,7 @@ check-disasm:
|
|||||||
$(PYTHON) dis-compare.py
|
$(PYTHON) dis-compare.py
|
||||||
|
|
||||||
#: Check deparsing bytecode 1.x only
|
#: Check deparsing bytecode 1.x only
|
||||||
check-bytecode-1: check-bytecode-1.4 check-bytecode-1.5
|
check-bytecode-1: check-bytecode-1.0 check-bytecode-1.1 check-bytecode-1.2 check-bytecode-1.3 check-bytecode-1.4 check-bytecode-1.5 check-bytecode-1.6
|
||||||
|
|
||||||
#: Check deparsing bytecode 2.x only
|
#: Check deparsing bytecode 2.x only
|
||||||
check-bytecode-2:
|
check-bytecode-2:
|
||||||
@@ -92,8 +98,9 @@ check-bytecode-2:
|
|||||||
check-bytecode-3:
|
check-bytecode-3:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.0 \
|
$(PYTHON) test_pythonlib.py --bytecode-3.0 \
|
||||||
--bytecode-3.1 --bytecode-3.2 --bytecode-3.3 \
|
--bytecode-3.1 --bytecode-3.2 --bytecode-3.3 \
|
||||||
--bytecode-3.4 --bytecode-3.5 --bytecode-3.6 --bytecode-3.7 \
|
--bytecode-3.4 --bytecode-3.5 --bytecode-3.6 \
|
||||||
--bytecode-pypy3.2
|
--bytecode-3.7 \
|
||||||
|
--bytecode-pypy3.2 --bytecode-pypy3.6 --bytecode-3.8
|
||||||
|
|
||||||
#: Check deparsing on selected bytecode 3.x
|
#: Check deparsing on selected bytecode 3.x
|
||||||
check-bytecode-3-short:
|
check-bytecode-3-short:
|
||||||
@@ -103,6 +110,7 @@ check-bytecode-3-short:
|
|||||||
#: Check deparsing bytecode on all Python 2 and Python 3 versions
|
#: Check deparsing bytecode on all Python 2 and Python 3 versions
|
||||||
check-bytecode: check-bytecode-3
|
check-bytecode: check-bytecode-3
|
||||||
$(PYTHON) test_pythonlib.py \
|
$(PYTHON) test_pythonlib.py \
|
||||||
|
--bytecode-1.0 --bytecode-1.1 --bytecode-1.2 \
|
||||||
--bytecode-1.3 --bytecode-1.4 --bytecode-1.5 \
|
--bytecode-1.3 --bytecode-1.4 --bytecode-1.5 \
|
||||||
--bytecode-2.2 --bytecode-2.3 --bytecode-2.4 \
|
--bytecode-2.2 --bytecode-2.3 --bytecode-2.4 \
|
||||||
--bytecode-2.1 --bytecode-2.2 --bytecode-2.3 --bytecode-2.4 \
|
--bytecode-2.1 --bytecode-2.2 --bytecode-2.3 --bytecode-2.4 \
|
||||||
@@ -116,6 +124,18 @@ check-bytecode-short: check-bytecode-3-short
|
|||||||
--bytecode-2.6 --bytecode-2.7 --bytecode-pypy2.7
|
--bytecode-2.6 --bytecode-2.7 --bytecode-pypy2.7
|
||||||
|
|
||||||
|
|
||||||
|
#: Check deparsing bytecode 1.0 only
|
||||||
|
check-bytecode-1.0:
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-1.0
|
||||||
|
|
||||||
|
#: Check deparsing bytecode 1.1 only
|
||||||
|
check-bytecode-1.1:
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-1.1
|
||||||
|
|
||||||
|
#: Check deparsing bytecode 1.2 only
|
||||||
|
check-bytecode-1.2:
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-1.2
|
||||||
|
|
||||||
#: Check deparsing bytecode 1.3 only
|
#: Check deparsing bytecode 1.3 only
|
||||||
check-bytecode-1.3:
|
check-bytecode-1.3:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-1.3
|
$(PYTHON) test_pythonlib.py --bytecode-1.3
|
||||||
@@ -128,6 +148,10 @@ check-bytecode-1.4:
|
|||||||
check-bytecode-1.5:
|
check-bytecode-1.5:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-1.5
|
$(PYTHON) test_pythonlib.py --bytecode-1.5
|
||||||
|
|
||||||
|
#: Check deparsing bytecode 1.6 only
|
||||||
|
check-bytecode-1.6:
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-1.6
|
||||||
|
|
||||||
#: Check deparsing Python 2.1
|
#: Check deparsing Python 2.1
|
||||||
check-bytecode-2.1:
|
check-bytecode-2.1:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-2.1
|
$(PYTHON) test_pythonlib.py --bytecode-2.1
|
||||||
@@ -170,7 +194,7 @@ grammar-coverage-2.6:
|
|||||||
grammar-coverage-2.7:
|
grammar-coverage-2.7:
|
||||||
-rm $(COVER_DIR)/spark-grammar-2.7.cover || true
|
-rm $(COVER_DIR)/spark-grammar-2.7.cover || true
|
||||||
SPARK_PARSER_COVERAGE=$(COVER_DIR)/spark-grammar-2.7.cover $(PYTHON) test_pythonlib.py --bytecode-2.7
|
SPARK_PARSER_COVERAGE=$(COVER_DIR)/spark-grammar-2.7.cover $(PYTHON) test_pythonlib.py --bytecode-2.7
|
||||||
SPARK_PARSER_COVERAGE=$(COVER_DIR)/spark-grammar-2.7.cover $(PYTHON) test_pyenvlib.py --2.7.14 --max=600
|
SPARK_PARSER_COVERAGE=$(COVER_DIR)/spark-grammar-2.7.cover $(PYTHON) test_pyenvlib.py --2.7.16 --max=600
|
||||||
|
|
||||||
#: Get grammar coverage for Python 3.0
|
#: Get grammar coverage for Python 3.0
|
||||||
grammar-coverage-3.0:
|
grammar-coverage-3.0:
|
||||||
@@ -213,77 +237,88 @@ grammar-coverage-3.5:
|
|||||||
grammar-coverage-3.6:
|
grammar-coverage-3.6:
|
||||||
rm $(COVER_DIR)/spark-grammar-3.6.cover || /bin/true
|
rm $(COVER_DIR)/spark-grammar-3.6.cover || /bin/true
|
||||||
SPARK_PARSER_COVERAGE=$(COVER_DIR)/spark-grammar-3.6.cover $(PYTHON) test_pythonlib.py --bytecode-3.6
|
SPARK_PARSER_COVERAGE=$(COVER_DIR)/spark-grammar-3.6.cover $(PYTHON) test_pythonlib.py --bytecode-3.6
|
||||||
SPARK_PARSER_COVERAGE=$(COVER_DIR)/spark-grammar-3.6.cover $(PYTHON) test_pyenvlib.py --3.6.4 --max=280
|
SPARK_PARSER_COVERAGE=$(COVER_DIR)/spark-grammar-3.6.cover $(PYTHON) test_pyenvlib.py --3.6.8 --max=280
|
||||||
|
|
||||||
|
#: Get grammar coverage for Python 3.7
|
||||||
|
grammar-coverage-3.7:
|
||||||
|
rm $(COVER_DIR)/spark-grammar-3.7.cover || /bin/true
|
||||||
|
SPARK_PARSER_COVERAGE=$(COVER_DIR)/spark-grammar-3.7.cover $(PYTHON) test_pyenvlib.py --3.7.3 --max=500
|
||||||
|
|
||||||
#: Check deparsing Python 2.6
|
#: Check deparsing Python 2.6
|
||||||
check-bytecode-2.6:
|
check-bytecode-2.6:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-2.6 --weak-verify
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-2.6-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-2.6-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-2.6 --syntax-verify
|
||||||
|
|
||||||
#: Check deparsing Python 2.7
|
#: Check deparsing Python 2.7
|
||||||
check-bytecode-2.7:
|
check-bytecode-2.7:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-2.7 --weak-verify
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-2.7-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-2.7-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-2.7 --syntax-verify
|
||||||
|
|
||||||
#: Check deparsing Python 3.0
|
#: Check deparsing Python 3.0
|
||||||
check-bytecode-3.0:
|
check-bytecode-3.0:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.0 --weak-verify
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.0-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.0-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.0 --syntax-verify
|
||||||
|
|
||||||
#: Check deparsing Python 3.1
|
#: Check deparsing Python 3.1
|
||||||
check-bytecode-3.1:
|
check-bytecode-3.1:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.1 --weak-verify
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.1-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.1-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.1 --syntax-verify
|
||||||
|
|
||||||
#: Check deparsing Python 3.2
|
#: Check deparsing Python 3.2
|
||||||
check-bytecode-3.2:
|
check-bytecode-3.2:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.2 --weak-verify
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.2-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.2-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.2 --syntax-verify
|
||||||
|
|
||||||
#: Check deparsing Python 3.3
|
#: Check deparsing Python 3.3
|
||||||
check-bytecode-3.3:
|
check-bytecode-3.3:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.3 --weak-verify
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.3-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.3-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.3 --syntax-verify
|
||||||
|
|
||||||
#: Check deparsing Python 3.4
|
#: Check deparsing Python 3.4
|
||||||
check-bytecode-3.4:
|
check-bytecode-3.4:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.4 --weak-verify
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.4-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.4-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.4 --syntax-verify
|
||||||
|
|
||||||
#: Check deparsing Python 3.5
|
#: Check deparsing Python 3.5
|
||||||
check-bytecode-3.5:
|
check-bytecode-3.5:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.5 --weak-verify
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.5-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.5-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.5 --syntax-verify
|
||||||
|
|
||||||
#: Check deparsing Python 3.6
|
#: Check deparsing Python 3.6
|
||||||
check-bytecode-3.6:
|
check-bytecode-3.6:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.6 --weak-verify
|
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.6-run --verify-run
|
$(PYTHON) test_pythonlib.py --bytecode-3.6-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.6 --syntax-verify
|
||||||
|
|
||||||
#: Check deparsing Python 3.7
|
#: Check deparsing Python 3.7
|
||||||
check-bytecode-3.7:
|
check-bytecode-3.7:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-3.7 --weak-verify
|
$(PYTHON) test_pythonlib.py --bytecode-3.7-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.7 --syntax-verify
|
||||||
|
|
||||||
|
#: Check deparsing Python 3.8
|
||||||
|
check-bytecode-3.8:
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.8-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-3.8 --syntax-verify
|
||||||
|
|
||||||
#: short tests for bytecodes only for this version of Python
|
#: short tests for bytecodes only for this version of Python
|
||||||
check-native-short:
|
check-native-short:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-$(PYTHON_VERSION) --weak-verify $(COMPILE)
|
$(PYTHON) test_pythonlib.py --bytecode-$(PYTHON_VERSION) --syntax-verify $(COMPILE)
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-$(PYTHON_VERSION)-run --verify-run $(COMPILE)
|
$(PYTHON) test_pythonlib.py --bytecode-$(PYTHON_VERSION)-run --verify-run $(COMPILE)
|
||||||
|
|
||||||
#: Run longer Python 2.6's lib files known to be okay
|
#: Run longer Python 2.6's lib files known to be okay
|
||||||
check-2.6-ok:
|
check-2.6-ok:
|
||||||
$(PYTHON) test_pythonlib.py --ok-2.6 --weak-verify $(COMPILE)
|
$(PYTHON) test_pythonlib.py --ok-2.6 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
#: Run longer Python 2.7's lib files known to be okay
|
#: Run longer Python 2.7's lib files known to be okay
|
||||||
check-2.7-ok:
|
check-2.7-ok:
|
||||||
$(PYTHON) test_pythonlib.py --ok-2.7 --weak-verify $(COMPILE)
|
$(PYTHON) test_pythonlib.py --ok-2.7 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
#: Run longer Python 3.2's lib files known to be okay
|
#: Run longer Python 3.2's lib files known to be okay
|
||||||
check-3.2-ok:
|
check-3.2-ok:
|
||||||
$(PYTHON) test_pythonlib.py --ok-3.2 --weak-verify $(COMPILE)
|
$(PYTHON) test_pythonlib.py --ok-3.2 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
#: Run longer Python 3.4's lib files known to be okay
|
#: Run longer Python 3.4's lib files known to be okay
|
||||||
check-3.4-ok:
|
check-3.4-ok:
|
||||||
$(PYTHON) test_pythonlib.py --ok-3.4 --weak-verify $(COMPILE)
|
$(PYTHON) test_pythonlib.py --ok-3.4 --syntax-verify $(COMPILE)
|
||||||
|
|
||||||
#: PyPy of some sort. E.g. [PyPy 5.0.1 with GCC 4.8.4]
|
#: PyPy of some sort. E.g. [PyPy 5.0.1 with GCC 4.8.4]
|
||||||
# Skip for now
|
# Skip for now
|
||||||
@@ -297,6 +332,20 @@ pypy-2.7 5.0 5.3 6.0:
|
|||||||
pypy-3.2 2.4:
|
pypy-3.2 2.4:
|
||||||
$(PYTHON) test_pythonlib.py --bytecode-pypy3.2 --verify
|
$(PYTHON) test_pythonlib.py --bytecode-pypy3.2 --verify
|
||||||
|
|
||||||
|
#: PyPy 5.0.x with Python 3.6.1 ...
|
||||||
|
check-bytecode-pypy3.6: 7.1
|
||||||
|
7.1:
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-pypy3.6-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-pypy3.6 --verify
|
||||||
|
|
||||||
|
#: PyPy 5.0.x with Python 3.6.9
|
||||||
|
check-bytecode-pypy3.6: 7.2
|
||||||
|
7.2:
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-pypy3.6-run --verify-run
|
||||||
|
$(PYTHON) test_pythonlib.py --bytecode-pypy3.6 --verify
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
clean: clean-py-dis clean-dis clean-unverified
|
clean: clean-py-dis clean-dis clean-unverified
|
||||||
|
|
||||||
clean-dis:
|
clean-dis:
|
||||||
|
@@ -2,15 +2,23 @@
|
|||||||
""" Trivial helper program to bytecompile and run an uncompile
|
""" Trivial helper program to bytecompile and run an uncompile
|
||||||
"""
|
"""
|
||||||
import os, sys, py_compile
|
import os, sys, py_compile
|
||||||
|
|
||||||
assert len(sys.argv) >= 2
|
assert len(sys.argv) >= 2
|
||||||
version = sys.version[0:3]
|
version = sys.version[0:3]
|
||||||
for path in sys.argv[1:]:
|
if sys.argv[1] in ("--run", "-r"):
|
||||||
|
suffix = "_run"
|
||||||
|
py_source = sys.argv[2:]
|
||||||
|
else:
|
||||||
|
suffix = ""
|
||||||
|
py_source = sys.argv[1:]
|
||||||
|
|
||||||
|
for path in py_source:
|
||||||
short = os.path.basename(path)
|
short = os.path.basename(path)
|
||||||
if hasattr(sys, 'pypy_version_info'):
|
if hasattr(sys, "pypy_version_info"):
|
||||||
cfile = "bytecode_pypy%s/%s" % (version, short) + 'c'
|
cfile = "bytecode_pypy%s%s/%s" % (version, suffix, short) + "c"
|
||||||
else:
|
else:
|
||||||
cfile = "bytecode_%s/%s" % (version, short) + 'c'
|
cfile = "bytecode_%s%s/%s" % (version, suffix, short) + "c"
|
||||||
print("byte-compiling %s to %s" % (path, cfile))
|
print("byte-compiling %s to %s" % (path, cfile))
|
||||||
py_compile.compile(path, cfile)
|
py_compile.compile(path, cfile)
|
||||||
if isinstance(version, str) or version >= (2, 6, 0):
|
if isinstance(version, str) or version >= (2, 6, 0):
|
||||||
os.system("../bin/uncompyle6 -a -t %s" % cfile)
|
os.system("../bin/uncompyle6 -a -T %s" % cfile)
|
||||||
|
BIN
test/bytecode_1.0/simple_const.pyc
Normal file
BIN
test/bytecode_1.0/simple_const.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_1.0/unpack_assign.pyc
Normal file
BIN
test/bytecode_1.0/unpack_assign.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_1.1/simple_const.pyc
Normal file
BIN
test/bytecode_1.1/simple_const.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_1.2/simple_const.pyc
Normal file
BIN
test/bytecode_1.2/simple_const.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_1.3/simple_const.pyc
Normal file
BIN
test/bytecode_1.3/simple_const.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_1.6/simple_const.pyc
Normal file
BIN
test/bytecode_1.6/simple_const.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_2.4/02_except_as.pyc
Normal file
BIN
test/bytecode_2.4/02_except_as.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_2.4_run/00_docstring.pyc
Normal file
BIN
test/bytecode_2.4_run/00_docstring.pyc
Normal file
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_2.6_run/00_generator.pyc
Normal file
BIN
test/bytecode_2.6_run/00_generator.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_2.6_run/01_if_while_return.pyc
Normal file
BIN
test/bytecode_2.6_run/01_if_while_return.pyc
Normal file
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_2.6_run/01_triple_compare.pyc
Normal file
BIN
test/bytecode_2.6_run/01_triple_compare.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_2.6_run/02_ifelse_comprehension.pyc
Normal file
BIN
test/bytecode_2.6_run/02_ifelse_comprehension.pyc
Normal file
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_2.6_run/04_and_del.pyc
Normal file
BIN
test/bytecode_2.6_run/04_and_del.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_2.6_run/04_ifelse_parens.pyc
Normal file
BIN
test/bytecode_2.6_run/04_ifelse_parens.pyc
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_2.7_run/00_docstring.pyc
Normal file
BIN
test/bytecode_2.7_run/00_docstring.pyc
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_2.7_run/01_triple_compare.pyc
Normal file
BIN
test/bytecode_2.7_run/01_triple_compare.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_2.7_run/02_ifelse_comprehension.pyc
Normal file
BIN
test/bytecode_2.7_run/02_ifelse_comprehension.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_2.7_run/02_slice.pyc
Normal file
BIN
test/bytecode_2.7_run/02_slice.pyc
Normal file
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_2.7_run/04_ifelse_parens.pyc
Normal file
BIN
test/bytecode_2.7_run/04_ifelse_parens.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_2.7_run/15_mixed_expressions.pyc
Normal file
BIN
test/bytecode_2.7_run/15_mixed_expressions.pyc
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_3.0/03_ifelse.pyc
Normal file
BIN
test/bytecode_3.0/03_ifelse.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.0/07_withstmt_fn.pyc
Normal file
BIN
test/bytecode_3.0/07_withstmt_fn.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.0_run/04_and_del.pyc
Normal file
BIN
test/bytecode_3.0_run/04_and_del.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.0_run/04_def_annotate.pyc
Normal file
BIN
test/bytecode_3.0_run/04_def_annotate.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.0_run/15_mixed_expressions.pyc
Normal file
BIN
test/bytecode_3.0_run/15_mixed_expressions.pyc
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_3.1_run/02_ifelse_comprehension.pyc
Normal file
BIN
test/bytecode_3.1_run/02_ifelse_comprehension.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.1_run/04_def_annotate.pyc
Normal file
BIN
test/bytecode_3.1_run/04_def_annotate.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.1_run/15_mixed_expressions.pyc
Normal file
BIN
test/bytecode_3.1_run/15_mixed_expressions.pyc
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_3.2_run/01_triple_compare.pyc
Normal file
BIN
test/bytecode_3.2_run/01_triple_compare.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.2_run/02_slice.pyc
Normal file
BIN
test/bytecode_3.2_run/02_slice.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.2_run/04_def_annotate.pyc
Normal file
BIN
test/bytecode_3.2_run/04_def_annotate.pyc
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_3.3/09_yield_from.pyc
Normal file
BIN
test/bytecode_3.3/09_yield_from.pyc
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_3.3_run/01_triple_compare.pyc
Normal file
BIN
test/bytecode_3.3_run/01_triple_compare.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.3_run/02_ifelse_comprehension.pyc
Normal file
BIN
test/bytecode_3.3_run/02_ifelse_comprehension.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.3_run/02_slice.pyc
Normal file
BIN
test/bytecode_3.3_run/02_slice.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.3_run/04_def_annotate.pyc
Normal file
BIN
test/bytecode_3.3_run/04_def_annotate.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.3_run/08_if_else.pyc
Normal file
BIN
test/bytecode_3.3_run/08_if_else.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.3_run/08_while_if.pyc
Normal file
BIN
test/bytecode_3.3_run/08_while_if.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.3_run/10_for.pyc
Normal file
BIN
test/bytecode_3.3_run/10_for.pyc
Normal file
Binary file not shown.
BIN
test/bytecode_3.3_run/10_mixed_boolean.pyc
Normal file
BIN
test/bytecode_3.3_run/10_mixed_boolean.pyc
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_3.4/09_yield_from.pyc
Normal file
BIN
test/bytecode_3.4/09_yield_from.pyc
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
test/bytecode_3.4_run/02_slice.pyc
Normal file
BIN
test/bytecode_3.4_run/02_slice.pyc
Normal file
Binary file not shown.
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user