python3生成.so动态库&执行exec函数的兼容问题

通过生成.so文件可以加密python项目,破解难度较大,但是会有代码兼容问题,并且在新系统下需要重新编译

整理一些python中执行exec函数编译为.so动态库之后碰到的问题

测试代码执行环境:

centos:centos7 镜像
Python 3.6.8
gcc version 4.8.5 20150623 (Red Hat 4.8.5-44) (GCC)

因为我这用的空的centos7环境所以安装东西比较多,正常比如使用macOS可能只需要安装Cython就可以了
安装:

yum -y install python3
curl https://bootstrap.pypa.io/get-pip.py -o get-pip.py
python3 get-pip.py
pip3 install Cython
yum -y install gcc
yum -y install python3-devel

生成.so的代码(gen-so.py):

#!/usr/bin/env python3

import sys 
from distutils.core import setup
from Cython.Build import cythonize

setup(ext_modules = cythonize([sys.argv[1]], compiler_directives={'language_level' : "3"}), script_args=['build_ext', '-b', './build', '-t', './tmp'])

举一个简单的例子:
python文件(demo.py):

def run():
    print('run')

生成.so文件, 运行命令:

python3 gen-so.py demo.py

生成后文件结构:

.
|-- build
|   `-- demo.cpython-36m-x86_64-linux-gnu.so
|-- demo.c
|-- demo.py
|-- gen-so.py
`-- tmp
    `-- demo.o

2 directories, 5 files

调用.so文件,这里方便起见就直接通过python -c调用了:

python3 -c "from build import demo;demo.run()"

会输出"run",没问题

调用exec函数编译为动态库后表现不一致的情况

1. exec中使用实例变量
demo.py

class Demo():
    def __init__(self):
        self.a = 1
        exec('self.b=2')
        print(self.b)    

def run():
    Demo()
    

生成.so文件后,分别执行Python和.so的代码:

[root@6d819d145bf6 demo]# python3 -c "import demo; demo.run()"
2
[root@6d819d145bf6 demo]# python3 -c "from build import demo; demo.run()"
Traceback (most recent call last):
  File "<string>", line 1, in <module>
  File "demo.py", line 8, in demo.run
    Demo()
  File "demo.py", line 4, in demo.Demo.__init__
    exec('self.b="2"')
  File "<string>", line 1, in <module>
NameError: name 'self' is not defined

修改exec('self.b=2')self.__dict__['b'] = 2或者exec('self.b=2',locals())

2. exec中使用局部变量
demo.py

class Demo():
    def __init__(self):
        self.a = 1
        local_c = 'lc'
        exec('local_d = "ld"')
        print(locals())    

def run():
    Demo()

分别执行python和.so:

[root@6d819d145bf6 demo]# python3 -c "import demo; demo.run()"
{'local_c': 'lc', 'self': <demo.Demo object at 0x7f4d89fa42b0>, 'local_d': 'ld'}
[root@6d819d145bf6 demo]# python3 -c "from build import demo; demo.run()"
{'local_c': 'lc', 'self': <demo.Demo object at 0x7f8136f3c2e8>}

可以看到Python能拿到exec中的局部变量local_d,但是.so动态库中不能拿到

3. exec中动态调用方法
demo.py

class Demo():
    def __init__(self):
        self.a = 1
        import math
        func_name = "log"
        exec("log10 = math."+ func_name + "(10)")
        print(log10)

def run():
    Demo()
    

编译失败:

[root@6d819d145bf6 demo]# python3 gen-so.py demo.py
Compiling demo.py because it changed.
[1/1] Cythonizing demo.py

Error compiling Cython file:
------------------------------------------------------------
...
    def __init__(self):
        self.a = 1
        import math
        func_name = "log"
        exec("log10 = math."+ func_name + "(10)")
        print(log10)
             ^
------------------------------------------------------------

demo.py:7:14: undeclared name not builtin: log10
Traceback (most recent call last):
  File "gen-so.py", line 7, in <module>
    setup(ext_modules = cythonize([sys.argv[1]], compiler_directives={'language_level' : "3"}), script_args=['build_ext', '-b', './build', '-t', './tmp'])
  File "/usr/local/lib64/python3.6/site-packages/Cython/Build/Dependencies.py", line 1102, in cythonize
    cythonize_one(*args)
  File "/usr/local/lib64/python3.6/site-packages/Cython/Build/Dependencies.py", line 1225, in cythonize_one
    raise CompileError(None, pyx_file)
Cython.Compiler.Errors.CompileError: demo.py

因为上面的log10是局部变量,所以通过locals()中也获取不到

修改exec("log10 = math."+ func_name + "(10)")log10 = getattr(math, func_name)(10)
直接运行.so:

[root@6d819d145bf6 demo]# python3 -c "from build import demo; demo.run()"
2.302585092994046

运行正确

4. exec代码内list使用局部变量
参考: listcomp unable to access locals defined in code called by exec if nested in function

这段代码在python和.so方式下都正常执行:

code = """
g = 5
x = [g for i in range(5)]
print(x)
"""
exec(code)

def run():
    pass

执行后正确输出[5, 5, 5, 5, 5]

下面这段代码在Python版执行报错,但是.so动态库执行成功:

def run():
    code = """
g = 5
x = [g for i in range(5)]
print(x)
"""
    exec(code)

python版报错,.so执行成功:

[root@6d819d145bf6 demo]# python3 -c "import demo; demo.run()"
Traceback (most recent call last):
  File "<string>", line 1, in <module>
  File "/opt/demo/demo.py", line 7, in run
    exec(code)
  File "<string>", line 3, in <module>
  File "<string>", line 3, in <listcomp>
NameError: name 'g' is not defined
[root@6d819d145bf6 demo]# python3 -c "from build import demo; demo.run()"
[5, 5, 5, 5, 5]

修改exec(code)exec(code, globals())后,python也运行正确,和.so表现一致

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,839评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,543评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,116评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,371评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,384评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,111评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,416评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,053评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,558评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,007评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,117评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,756评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,324评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,315评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,539评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,578评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,877评论 2 345

推荐阅读更多精彩内容