Skip to content

Commit 5e37435

Browse files
committed
Nicer __main__ support for Python 3
ISSUE: DEVTOOLS-4841 ref:d402badab719bf5a7bb3aef4e470723ba1a709aa
1 parent 974c988 commit 5e37435

File tree

4 files changed

+70
-33
lines changed

4 files changed

+70
-33
lines changed

build/plugins/pybuild.py

+12-9
Original file line numberDiff line numberDiff line change
@@ -215,22 +215,19 @@ def onpy_srcs(unit, *args):
215215
# Unsupported but legal PROTO_LIBRARY arguments.
216216
elif arg == 'GLOBAL' or arg.endswith('.gztproto'):
217217
pass
218-
# XXX https://st.yandex-team.ru/DEVTOOLS-4841
219-
elif py3 and (arg == '__main__.py' or arg.endswith('/__main__.py')):
220-
unit.onfix_python_main([arg, '__real_main__.py'])
221-
unit.onpy_srcs(['TOP_LEVEL', '__real_main__.py'])
222-
unit.onpy_main(['__real_main__:real_main_func'])
223218
# Sources.
224219
else:
225220
main_mod = arg == 'MAIN'
226221
if main_mod:
227222
arg = next(args)
228223

229224
if '=' in arg:
225+
main_py = False
230226
path, mod = arg.split('=', 1)
231227
else:
232228
path = arg
233-
if arg == '__main__.py' or arg.endswith('/__main__.py'):
229+
main_py = (path == '__main__.py' or path.endswith('/__main__.py'))
230+
if not py3 and main_py:
234231
mod = '__main__'
235232
else:
236233
if arg.startswith('../'):
@@ -241,7 +238,9 @@ def onpy_srcs(unit, *args):
241238
mod = ns + stripext(arg).replace('/', '.')
242239

243240
if main_mod:
244-
unit.onpy_main(mod)
241+
py_main(unit, mod + ":main")
242+
elif py3 and main_py:
243+
py_main(unit, mod)
245244

246245
pathmod = (path, mod)
247246

@@ -455,6 +454,11 @@ def onpy_register(unit, *args):
455454
py_register(unit, name, py3)
456455

457456

457+
def py_main(unit, arg):
458+
py_program(unit, is_py3(unit))
459+
unit.onresource(['-', 'PY_MAIN={}'.format(arg)])
460+
461+
458462
def onpy_main(unit, arg):
459463
"""
460464
@usage: PY_MAIN(pkg.mod[:func])
@@ -466,5 +470,4 @@ def onpy_main(unit, arg):
466470
if ':' not in arg:
467471
arg += ':main'
468472

469-
py_program(unit, is_py3(unit))
470-
unit.onresource(['-', 'PY_MAIN={}'.format(arg)])
473+
py_main(unit, arg)

build/scripts/fix_python_main.py

-11
This file was deleted.

build/ymake.core.conf

-4
Original file line numberDiff line numberDiff line change
@@ -1994,10 +1994,6 @@ macro COPY_FILE(File, Destination) {
19941994
.CMD=$COPY_CMD ${input:File} ${output;noauto:Destination} ${kv;hide:"p CP"} ${kv;hide:"pc light-cyan"}
19951995
}
19961996

1997-
macro FIX_PYTHON_MAIN(File, Destination) {
1998-
.CMD=$YMAKE_PYTHON ${input:"build/scripts/fix_python_main.py"} ${input:File} ${output;noauto:Destination} ${kv;hide:"p CP"} ${kv;hide:"pc light-cyan"}
1999-
}
2000-
20011997
macro BUNDLE_TARGET(Target, Destination) {
20021998
.CMD=$MOVE_FILE ${result:Target} ${output:Destination} ${kv;hide:"p BN"} ${kv;hide:"pc light-cyan"}
20031999
}

library/python/runtime_py3/main/main.c

+58-9
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,55 @@ void unsetenv(const char* name) {
1919
}
2020
#endif
2121

22+
static int RunModule(const char *modname)
23+
{
24+
PyObject *module, *runpy, *runmodule, *runargs, *result;
25+
runpy = PyImport_ImportModule("runpy");
26+
if (runpy == NULL) {
27+
fprintf(stderr, "Could not import runpy module\n");
28+
PyErr_Print();
29+
return -1;
30+
}
31+
runmodule = PyObject_GetAttrString(runpy, "_run_module_as_main");
32+
if (runmodule == NULL) {
33+
fprintf(stderr, "Could not access runpy._run_module_as_main\n");
34+
PyErr_Print();
35+
Py_DECREF(runpy);
36+
return -1;
37+
}
38+
module = PyUnicode_FromString(modname);
39+
if (module == NULL) {
40+
fprintf(stderr, "Could not convert module name to unicode\n");
41+
PyErr_Print();
42+
Py_DECREF(runpy);
43+
Py_DECREF(runmodule);
44+
return -1;
45+
}
46+
runargs = Py_BuildValue("(Oi)", module, 0);
47+
if (runargs == NULL) {
48+
fprintf(stderr,
49+
"Could not create arguments for runpy._run_module_as_main\n");
50+
PyErr_Print();
51+
Py_DECREF(runpy);
52+
Py_DECREF(runmodule);
53+
Py_DECREF(module);
54+
return -1;
55+
}
56+
result = PyObject_Call(runmodule, runargs, NULL);
57+
if (result == NULL) {
58+
PyErr_Print();
59+
}
60+
Py_DECREF(runpy);
61+
Py_DECREF(runmodule);
62+
Py_DECREF(module);
63+
Py_DECREF(runargs);
64+
if (result == NULL) {
65+
return -1;
66+
}
67+
Py_DECREF(result);
68+
return 0;
69+
}
70+
2271
int main(int argc, char** argv) {
2372
int i, sts = 1;
2473
char* oldloc = NULL;
@@ -111,12 +160,14 @@ int main(int argc, char** argv) {
111160
func_name = colon + 1;
112161
}
113162

114-
PyObject* module = PyImport_ImportModule(module_name);
115-
116-
if (module == NULL) {
117-
PyErr_Print();
163+
if (!func_name) {
164+
sts = RunModule(module_name);
118165
} else {
119-
if (func_name) {
166+
PyObject* module = PyImport_ImportModule(module_name);
167+
168+
if (module == NULL) {
169+
PyErr_Print();
170+
} else {
120171
PyObject* value = PyObject_CallMethod(module, func_name, NULL);
121172

122173
if (value == NULL) {
@@ -125,11 +176,9 @@ int main(int argc, char** argv) {
125176
Py_DECREF(value);
126177
sts = 0;
127178
}
128-
} else {
129-
sts = 0;
130-
}
131179

132-
Py_DECREF(module);
180+
Py_DECREF(module);
181+
}
133182
}
134183

135184
if (Py_FinalizeEx() < 0) {

0 commit comments

Comments
 (0)