generated from xzhuah/BlenderAddonPackageTool
-
Notifications
You must be signed in to change notification settings - Fork 0
/
framework.py
683 lines (578 loc) · 29 KB
/
framework.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
import ast
import atexit
import os
import re
import shutil
import subprocess
import sys
import threading
import time
from datetime import datetime
from pathlib import Path
from common.class_loader.module_installer import install_if_missing, install_fake_bpy
from common.io.FileManagerClient import search_files, read_utf8, write_utf8, is_subdirectory, get_md5_folder, \
read_utf8_in_lines, write_utf8_in_lines
from main import PROJECT_ROOT, BLENDER_ADDON_PATH, BLENDER_EXE_PATH, DEFAULT_RELEASE_DIR, TEST_RELEASE_DIR, IS_EXTENSION
try:
# added in python3.11
import tomllib
except ImportError:
# for python3.10 and below
install_if_missing("toml")
import toml
# Following variables are used internally in the framework according to some protocols defined by Blender or
# the framework itself. Do not change them unless you know what you are doing.
_addon_namespace_pattern = re.compile("^[a-zA-Z]+[a-zA-Z0-9_]*$")
_import_module_pattern = re.compile("from ([a-zA-Z_][a-zA-Z0-9_.]*) import (.+)")
_relative_import_pattern = re.compile(r'^\s*(from\s+(\.+))(.*)$')
_absolute_import_pattern = re.compile(r'^\s*from\s+(\w+[\w.]*)\s+import\s+(.*)$')
_addon_md5__signature = "addon.txt"
_ADDON_MANIFEST_FILE = "blender_manifest.toml"
_WHEELS_PATH = "wheels"
# 默认使用的插件模板 不要轻易修改
_ADDON_TEMPLATE = "sample_addon"
_ADDONS_FOLDER = "addons"
_ADDON_ROOT = os.path.join(PROJECT_ROOT, _ADDONS_FOLDER)
# Install fake bpy module only when user have configured the blender executable path
# 仅在用户配置了Blender可执行文件路径时安装fake bpy模块 避免在非Blender环境下安装fake bpy模块(如CICD流程中)
if os.path.isfile(BLENDER_EXE_PATH):
install_fake_bpy(BLENDER_EXE_PATH)
def new_addon(addon_name: str):
new_addon_path = os.path.join(_ADDON_ROOT, addon_name)
if os.path.exists(new_addon_path):
raise ValueError("Addon already exists: " + addon_name)
if not bool(_addon_namespace_pattern.match(addon_name)):
raise ValueError("Invalid addon name: " + addon_name + " Please name it as a python package name")
shutil.copytree(os.path.join(_ADDON_ROOT, _ADDON_TEMPLATE), new_addon_path)
all_template_file = search_files(new_addon_path, {".py", ".toml"})
for py_file in all_template_file:
content = read_utf8(py_file).replace(_ADDON_TEMPLATE, addon_name)
write_utf8(py_file, content)
def test_addon(addon_name, enable_watch=True):
init_file = get_init_file_path(addon_name)
if not enable_watch:
print('Do not auto reload addon when file changed')
start_test(init_file, addon_name, enable_watch=enable_watch)
def get_init_file_path(addon_name):
# addon_name is the name defined in addon's config.py
target_init_file_path = os.path.join(_ADDON_ROOT, addon_name, "__init__.py")
if not os.path.exists(target_init_file_path):
raise ValueError(f"Release failed: Addon {addon_name} not found.")
return target_init_file_path
# The following code will be injected into the blender python environment to enable hot reload
# https://devtalk.blender.org/t/plugin-hot-reload-by-cleaning-sys-modules/20040
start_up_command = """
import bpy
from bpy.app.handlers import persistent
import os
import sys
existing_addon_md5 = ""
try:
bpy.ops.preferences.addon_enable(module="{addon_name}")
except Exception as e:
print("Addon enable failed:", e)
def watch_update_tick():
global existing_addon_md5
if os.path.exists("{addon_signature}"):
with open("{addon_signature}", "r") as f:
addon_md5 = f.read()
if existing_addon_md5 == "":
existing_addon_md5 = addon_md5
elif existing_addon_md5 != addon_md5:
print("Addon file changed, start to update the addon")
try:
bpy.ops.preferences.addon_disable(module="{addon_name}")
all_modules = sys.modules
all_modules = dict(sorted(all_modules.items(),key= lambda x:x[0])) #sort them
for k,v in all_modules.items():
if k.startswith("{addon_name}"):
del sys.modules[k]
bpy.ops.preferences.addon_enable(module="{addon_name}")
except Exception as e:
print("Addon update failed:", e)
existing_addon_md5 = addon_md5
print("Addon updated")
return 1.0
@persistent
def register_watch_update_tick(dummy):
print("Watching for addon update...")
bpy.app.timers.register(watch_update_tick)
register_watch_update_tick(None)
bpy.app.handlers.load_post.append(register_watch_update_tick)
"""
def start_test(init_file, addon_name, enable_watch=True):
update_addon_for_test(init_file, addon_name)
test_addon_path = os.path.normpath(os.path.join(BLENDER_ADDON_PATH, addon_name))
if not enable_watch:
def exit_handler():
if os.path.exists(test_addon_path):
shutil.rmtree(test_addon_path)
atexit.register(exit_handler)
try:
execute_blender_script(
[BLENDER_EXE_PATH, "--python-use-system-env", "--python-expr",
f"import bpy\nbpy.ops.preferences.addon_enable(module=\"{addon_name}\")"], test_addon_path)
finally:
exit_handler()
return
# start_watch_for_update(init_file, addon_name)
stop_event = threading.Event()
thread = threading.Thread(target=start_watch_for_update, args=(init_file, addon_name, stop_event))
thread.start()
def exit_handler():
stop_event.set()
thread.join()
if os.path.exists(test_addon_path):
shutil.rmtree(test_addon_path)
atexit.register(exit_handler)
python_script = start_up_command.format(addon_name=addon_name,
addon_signature=os.path.join(test_addon_path,
_addon_md5__signature).replace("\\", "/"))
try:
execute_blender_script([BLENDER_EXE_PATH, "--python-use-system-env", "--python-expr", python_script],
test_addon_path)
finally:
exit_handler()
# This is the only corner case need to handle
_addon_on_init_file = os.path.abspath(os.path.join(PROJECT_ROOT, "__init__.py"))
def execute_blender_script(args, addon_path):
process = subprocess.Popen(args, stderr=subprocess.PIPE, text=True, encoding="utf-8")
try:
for line in process.stderr:
line: str
if line.lstrip().startswith("File"):
line = line.replace(addon_path, PROJECT_ROOT)
sys.stderr.write(line)
except KeyboardInterrupt:
sys.stderr.write("interrupted, terminating the child process...\n")
finally:
process.terminate()
process.wait()
def read_ext_config(addon_config_file):
with open(addon_config_file, 'r', encoding='utf-8') as f:
try:
addon_config = tomllib.loads(f.read())
except Exception as e:
addon_config = toml.load(f)
return addon_config
def release_addon(target_init_file, addon_name,
release_dir=DEFAULT_RELEASE_DIR,
need_zip=True,
is_extension=IS_EXTENSION,
with_timestamp=False,
with_version=False):
# if release dir is under PROJECT_ROOT, it's not allowed
if is_subdirectory(release_dir, PROJECT_ROOT):
# 不要将插件发布目录设置在当前项目内
raise ValueError("Invalid release dir:", release_dir,
"Please set a release/test dir outside the current workspace")
if not bool(_addon_namespace_pattern.match(addon_name)):
raise ValueError("InValid addon_name:", addon_name, "Please name it as a python package name")
if is_extension:
# 发布为扩展时,请确保您在config.py正确的定义了__addon_name__
# Release as extension, please make sure you defined __addon_name__ correctly in config.py"
# Make sure toml file exists
addon_config_file = os.path.join(_ADDON_ROOT, addon_name, _ADDON_MANIFEST_FILE)
if not os.path.isfile(addon_config_file):
raise ValueError("Extension config file not found:", addon_config_file)
if not os.path.isdir(release_dir):
Path(release_dir).mkdir(parents=True, exist_ok=True)
# remove the folder if already exists
release_folder = os.path.join(release_dir, addon_name)
if os.path.exists(release_folder):
shutil.rmtree(release_folder)
os.mkdir(release_folder)
bootstrap_init_file = generate_bootstrap_init_file(addon_name, get_addon_info(target_init_file))
write_utf8(os.path.join(release_folder, "__init__.py"), bootstrap_init_file)
# shutil.copyfile(target_init_file, os.path.join(release_folder, "__init__.py"))
# 将target_init_file同级的其他非py文件复制到发布目录 如 toml xml等可能跟插件有关的配置文件
for file in os.listdir(os.path.dirname(target_init_file)):
file_path = os.path.join(os.path.dirname(target_init_file), file)
if os.path.isdir(file_path) or file.endswith(".py"):
continue
shutil.copy(file_path, release_folder)
# 将插件文件夹复制到发布目录
shutil.copytree(os.path.join(_ADDON_ROOT, addon_name), os.path.join(release_folder, _ADDONS_FOLDER, addon_name))
shutil.copyfile(os.path.join(_ADDON_ROOT, "__init__.py"),
os.path.join(release_folder, _ADDONS_FOLDER, "__init__.py"))
all_py_files = search_files(os.path.join(_ADDON_ROOT, addon_name), {".py"})
# 对插件文件夹中的每一个py文件进行分析,找到每个py文件中依赖的其他py文件
visited_py_files = set()
for py_file in all_py_files:
visited_py_files.add(os.path.abspath(py_file))
# 注意不要漏掉__init__.py文件
visited_py_files.add(os.path.abspath(os.path.join(_ADDON_ROOT, "__init__.py")))
dependencies = find_all_dependencies(list(visited_py_files), PROJECT_ROOT)
for dependency in dependencies:
dependency = os.path.abspath(dependency)
if dependency in visited_py_files:
continue
visited_py_files.add(dependency)
target_path = os.path.join(release_folder, os.path.relpath(dependency, PROJECT_ROOT))
if not os.path.exists(os.path.dirname(target_path)):
os.makedirs(os.path.dirname(target_path))
shutil.copy(dependency, os.path.join(release_folder, os.path.relpath(dependency, PROJECT_ROOT)))
remove_pyc_files(release_folder)
removed_path = 1
while removed_path > 0:
removed_path = remove_empty_folders(release_folder)
# 必须先将绝对导入转换为相对导入,否则enhance_import_for_py_files一步会改变绝对导入的路径导致出错
# convert absolute import to relative import if it's an extension
if is_extension:
for py_file in search_files(release_folder, {".py"}):
convert_absolute_to_relative(py_file, release_folder)
# 更新打包后的绝对导入路径:由于打包后文件夹的层级关系发生了变化,需要更新打包后的绝对导入路径
enhance_import_for_py_files(release_folder)
# enhance relative import for root __init__.py
# enhance_relative_import_for_init_py(os.path.join(release_folder, "__init__.py"),
# _ADDONS_FOLDER, addon_name)
# include wheel files when need to be zipped
addon_config_file = os.path.join(_ADDON_ROOT, addon_name, _ADDON_MANIFEST_FILE)
addon_config = {}
if os.path.exists(addon_config_file) and is_extension:
addon_config = read_ext_config(addon_config_file)
if need_zip:
# package whl files into extension
if "wheels" in addon_config:
wheel_files = addon_config["wheels"]
if len(wheel_files) > 0:
wheel_folder = os.path.join(release_folder, _WHEELS_PATH)
os.mkdir(wheel_folder)
for wheel_file in wheel_files:
# You much put the .whl file directly under the wheels folder, not in a subfolder
# 你必须将.whl文件直接放在wheels文件夹下,而不是在子文件夹中
assert wheel_file.startswith("./wheels/") and wheel_file.count("/") == 2
wheel_source = os.path.join(PROJECT_ROOT, wheel_file)
if not os.path.exists(wheel_source):
raise ValueError("Wheel file not found:", wheel_source,
". Please download the required wheel file to the wheels folder.")
shutil.copy(wheel_source, wheel_folder)
real_addon_name = "{addon_name}".format(addon_name=release_folder)
if is_extension:
real_addon_name = f"{real_addon_name}_ext"
if with_version:
_version: str
if not is_extension:
bl_info = get_addon_info(target_init_file)
if bl_info is not None:
_version = '.'.join([str(x) for x in bl_info['version']])
else:
raise ValueError("bl_info not found in:", target_init_file)
else:
if "version" in addon_config:
_version = addon_config["version"]
else:
raise ValueError("version not found in:", addon_config_file)
real_addon_name = f"{real_addon_name}_V{_version}"
if with_timestamp:
timestamp = datetime.now().strftime("%Y%m%d_%H%M%S")
real_addon_name = f"{real_addon_name}_{timestamp}"
released_addon_path = os.path.abspath(os.path.join(release_dir, real_addon_name) + ".zip")
# zip the addon
if need_zip:
zip_folder(release_folder, real_addon_name, is_extension)
print("Add on released:", released_addon_path)
return released_addon_path
def get_addon_info(filename: str):
file_content = read_utf8(filename)
try:
parsed_ast = ast.parse(file_content)
for node in ast.walk(parsed_ast):
if isinstance(node, ast.Assign):
for target in node.targets:
if isinstance(target, ast.Name) and target.id == "bl_info":
return ast.literal_eval(node.value)
except Exception as e:
return None
def generate_bootstrap_init_file(addon_name: str, bl_info: dict):
bootstrap_init_file_template = """from .addons.{addon_name} import register as addon_register, unregister as addon_unregister
bl_info = {bl_info}
def register():
addon_register()
def unregister():
addon_unregister()
"""
bl_info_str = (
"{\n"
+ ",\n".join(
f' "{key}": {repr(value)}'
for key, value in bl_info.items()
)
+ "\n}"
)
return bootstrap_init_file_template.format(addon_name=addon_name, bl_info=bl_info_str)
# pyc files are auto generated, need to be removed before release
def remove_pyc_files(release_folder: str):
all_pyc_file = search_files(release_folder, {"pyc"})
for pyc_file in all_pyc_file:
os.remove(pyc_file)
def remove_empty_folders(root_path):
all_folder_to_remove = []
for root, dirnames, filenames in os.walk(root_path, topdown=False):
for dirname in dirnames:
dir_to_check = os.path.join(root, dirname)
if not os.listdir(dir_to_check):
all_folder_to_remove.append(dir_to_check)
for folder in all_folder_to_remove:
shutil.rmtree(folder)
return len(all_folder_to_remove)
# Zip the folder in a way that blender can recognize it as an addon.
def zip_folder(target_root, output_zip_file, is_extension):
if is_extension:
shutil.make_archive(output_zip_file, 'zip', Path(target_root))
else:
shutil.make_archive(output_zip_file, 'zip', Path(target_root).parent, base_dir=os.path.basename(target_root))
def find_imported_modules(file_path):
root = ast.parse(read_utf8(file_path), filename=file_path)
imported_modules = set()
for node in ast.walk(root):
if isinstance(node, ast.Import):
for alias in node.names:
imported_modules.add(alias.name)
elif isinstance(node, ast.ImportFrom):
if node.module:
module_name = node.module
imported_modules.add(module_name)
for alias in node.names:
if node.module:
imported_modules.add(f"{node.module}.{alias.name}")
else:
imported_modules.add(alias.name)
return imported_modules
def resolve_module_path(module_name, base_path, project_root):
if not module_name.endswith(".*"):
# Handle import all
module_path = module_name.replace('.', '/')
module_path = os.path.join(project_root, module_path)
if os.path.isdir(module_path):
module_path = os.path.join(module_path, '__init__.py')
return [module_path]
elif os.path.isfile(module_path + '.py'):
module_path = module_path + '.py'
return [module_path]
else:
if "." not in module_name:
# most likely a standard library module
# 有一种可能是相对导入 from . import xxx, from .. import xxx 等
# 这种情况下需要根据当前文件的路径来解析 看module_name.py是否存在于当前文件的同级目录或者父级目录
# 从base_path开始向上查找,直到找到module_name.py或者到达project_root
search_path = os.path.dirname(base_path)
potential_result = []
while is_subdirectory(search_path, project_root):
possible_path = os.path.join(search_path, module_name + '.py')
if os.path.isfile(possible_path):
potential_result.append(possible_path)
search_path = os.path.dirname(search_path)
return potential_result
current_search_dir = os.path.dirname(base_path)
while is_subdirectory(current_search_dir, project_root):
module_path = module_name.replace('.', '/')
module_path = os.path.join(current_search_dir, module_path)
if os.path.isdir(module_path):
module_path = os.path.join(module_path, '__init__.py')
return [module_path]
elif os.path.isfile(module_path + '.py'):
module_path = module_path + '.py'
return [module_path]
current_search_dir = os.path.dirname(current_search_dir)
return []
else:
module_name = module_name[:-2]
module_path = module_name.replace('.', '/')
possible_root_path = os.path.join(project_root, module_path)
if os.path.isdir(possible_root_path):
possible_root_path = os.path.join(possible_root_path, '__init__.py')
return [possible_root_path]
elif os.path.isfile(possible_root_path + '.py'):
possible_root_path = possible_root_path + '.py'
return [possible_root_path]
else:
current_search_dir = os.path.dirname(base_path)
while is_subdirectory(current_search_dir, project_root):
possible_root_path = os.path.join(current_search_dir, module_path)
if os.path.isdir(possible_root_path):
possible_root_path = os.path.join(possible_root_path, '__init__.py')
return [possible_root_path]
elif os.path.isfile(possible_root_path + '.py'):
possible_root_path = possible_root_path + '.py'
return [possible_root_path]
current_search_dir = os.path.dirname(current_search_dir)
return []
def find_all_dependencies(file_paths: list, project_root: str):
dependencies = set()
to_process = file_paths.copy()
processed = set()
while to_process:
current_file = os.path.abspath(to_process.pop())
if current_file in processed:
continue
processed.add(current_file)
dependencies.add(current_file)
try:
imported_modules = find_imported_modules(current_file)
except SyntaxError as e:
raise SyntaxError(f"Syntax error in file {current_file}: {e}")
# 以下代码会将除了当前目标插件文件夹以外的其他被引用的文件夹中的__init__.py文件也加入到依赖中,使之成为有效的模块,从而将其中的Blender
# 类也加入到自动注册的范围中,一般来说,我们引用外部文件夹的目的是复用其内部函数,而非将插件外部模块中定义的Operator,Panel等元素
# 直接加到当前插件中(如果需要使用其他插件的这些元素,更好的做法是将其直接存放到你的插件文件夹内),因此注释掉,如果您有特殊需求,可以取消注释
# The following code will add __init__.py files in other
# referenced folders to the dependencies, in addition to the current ACTIVE ADDON ,making those folders valid
# modules and thus classes in them will be added the scope of automatic class registration. (The
# auto_load.py) It is commented out because usually we just want to reference reusable functions from
# modules outside the current addon Instead of directly adding their Operator's Panels into your own addon. (
# If you really want to do that, include them as sub package of your own addon would be a better option). But
# If you have special requirements, you can uncomment it.
# potential_init_file = os.path.abspath(os.path.join(os.path.dirname(current_file), '__init__.py'))
# while is_subdirectory(os.path.dirname(potential_init_file),
# project_root) and potential_init_file != os.path.abspath(
# os.path.join(project_root, "__init__.py")):
# if os.path.exists(potential_init_file) and potential_init_file not in processed:
# to_process.append(potential_init_file)
# dependencies.add(potential_init_file)
# potential_init_file = os.path.abspath(
# os.path.join(os.path.dirname(os.path.dirname(potential_init_file)), '__init__.py'))
for module in imported_modules:
module_path = resolve_module_path(module, current_file, project_root)
if len(module_path) > 0:
for each_module_path in module_path:
each_module_path = os.path.abspath(each_module_path)
if each_module_path not in processed:
to_process.append(each_module_path)
return dependencies
def enhance_import_for_py_files(addon_dir: str):
namespace = os.path.basename(addon_dir)
all_py_modules = find_all_py_modules(addon_dir)
all_py_file = search_files(addon_dir, {".py"})
for py_file in all_py_file:
hasUpdated = False
content = read_utf8(py_file)
for module_path in _import_module_pattern.finditer(content):
original_module_path = module_path.groups()[0]
if original_module_path in all_py_modules:
hasUpdated = True
content = content.replace("from " + original_module_path + " import",
"from " + namespace + "." + original_module_path + " import")
if hasUpdated:
write_utf8(py_file, content)
def convert_absolute_to_relative(file_path: str, project_root: str):
"""
Convert all absolute imports to relative imports in a Python file.
Notice this does not handle import like
import xxx.yyy.zzz as zzz
在开发扩展时,不要用这种方式导入项目内的模块,这种方式导入的模块无法被转换为相对导入
Args:
file_path (str): Path to the Python file to modify.
project_root (str): Root directory of the project.
"""
# Normalize paths
file_path = os.path.abspath(file_path)
project_root = os.path.abspath(project_root)
lines = read_utf8_in_lines(file_path)
modified_lines = []
changed = False
for line in lines:
# help skipping expensive path check
stripped_line = line.strip()
if (not stripped_line.startswith("from ")) or stripped_line.startswith("from ."):
# Leave non-import lines unchanged
modified_lines.append(line)
continue
match = _absolute_import_pattern.match(line)
if match:
# get whitespace before the import statement
leading_space = line[:line.index("from")]
absolute_module = match.group(1)
import_items = match.group(2)
# Check if the absolute module is within the project
absolute_module_path = absolute_module.replace('.', os.sep)
full_module_path = os.path.join(project_root, absolute_module_path)
if os.path.exists(full_module_path) or os.path.exists(f"{full_module_path}.py"):
# Calculate the relative import path
target_relative_path = os.path.relpath(
os.path.join(project_root, absolute_module_path),
os.path.dirname(file_path)
)
# Count the levels for leading dots
levels_up = target_relative_path.count("..") + 1
leading_dots = '.' * levels_up
# Build the relative import line
target_relative_path = target_relative_path.strip("." + os.sep)
relative_import_line = leading_space + f"from {leading_dots}{target_relative_path.replace(os.sep, '.')} import {import_items}\n"
if relative_import_line != line:
modified_lines.append(relative_import_line)
changed = True
continue
else:
# Leave non-matching lines unchanged
modified_lines.append(line)
else:
# Leave non-matching lines unchanged
modified_lines.append(line)
# print(f"not match {line} in {timer() - start3} seconds")
# Write the modified content back to the file if changes were made
if changed:
write_utf8_in_lines(file_path, modified_lines)
def find_all_py_modules(root_dir: str) -> set:
all_py_modules = set()
all_py_file = search_files(root_dir, {".py"})
for py_file in all_py_file:
rel_path = str(os.path.relpath(py_file, root_dir))
modules = rel_path.replace("__init__.py", "").replace(".py", "").split(os.path.sep)
if len(modules[-1]) == 0:
modules = modules[0:-1]
module_name = ""
for i in range(len(modules)):
module_name += modules[i] + "."
all_py_modules.add(module_name[0:-1])
return all_py_modules
def start_watch_for_update(init_file, addon_name, stop_event: threading.Event):
install_if_missing("watchdog")
from watchdog.events import FileSystemEventHandler
from watchdog.observers import Observer
class FileUpdateHandler(FileSystemEventHandler):
def __init__(self):
super(FileUpdateHandler, self).__init__()
self.has_update = False
def on_any_event(self, event):
source_path = event.src_path
if source_path.endswith(".py"):
self.has_update = True
def clear_update(self):
self.has_update = False
path = PROJECT_ROOT
event_handler = FileUpdateHandler()
observer = Observer()
observer.schedule(event_handler, path, recursive=True)
observer.start()
try:
while not stop_event.is_set():
time.sleep(1)
if event_handler.has_update:
try:
update_addon_for_test(init_file, addon_name)
event_handler.clear_update()
except Exception as e:
print(e)
print(
"Addon updated failed: Please make sure no other process is"
" using the addon folder. You might need to restart the test to update the addon in Blender.")
print("Stop watching for update...")
except KeyboardInterrupt:
observer.stop()
observer.join()
def update_addon_for_test(init_file, addon_name):
if BLENDER_ADDON_PATH is None:
# 无法得到Blender插件路径 请检查在main.py或config.ini中的配置
raise ValueError(
"Could not find Blender addon installation path. Please check the configuration in main.py or config.ini")
addon_path = release_addon(init_file, addon_name, with_timestamp=False,
is_extension=IS_EXTENSION,
release_dir=TEST_RELEASE_DIR, need_zip=False)
executable_path = os.path.join(os.path.dirname(addon_path), addon_name)
test_addon_path = os.path.join(BLENDER_ADDON_PATH, addon_name)
if os.path.exists(test_addon_path):
shutil.rmtree(test_addon_path)
shutil.copytree(executable_path, test_addon_path)
# write an MD5 to the addon folder to inform the addon content has been changed
addon_md5 = get_md5_folder(executable_path)
write_utf8(os.path.join(test_addon_path, _addon_md5__signature), addon_md5)