Deanna Garcia | 859410b | 2023-05-01 15:11:07 -0700 | [diff] [blame] | 1 | """ |
| 2 | Internal helpers for building the Python protobuf runtime. |
| 3 | """ |
| 4 | |
| 5 | def _remove_cross_repo_path(path): |
| 6 | components = path.split("/") |
| 7 | if components[0] == "..": |
| 8 | return "/".join(components[2:]) |
| 9 | return path |
David L. Jones | b3cbea1 | 2022-05-12 19:48:58 -0700 | [diff] [blame] | 10 | |
| 11 | def _internal_copy_files_impl(ctx): |
| 12 | strip_prefix = ctx.attr.strip_prefix |
| 13 | if strip_prefix[-1] != "/": |
| 14 | strip_prefix += "/" |
| 15 | |
| 16 | src_dests = [] |
| 17 | for src in ctx.files.srcs: |
Deanna Garcia | 859410b | 2023-05-01 15:11:07 -0700 | [diff] [blame] | 18 | short_path = _remove_cross_repo_path(src.short_path) |
| 19 | if short_path[:len(strip_prefix)] != strip_prefix: |
David L. Jones | b3cbea1 | 2022-05-12 19:48:58 -0700 | [diff] [blame] | 20 | fail("Source does not start with %s: %s" % |
Deanna Garcia | 859410b | 2023-05-01 15:11:07 -0700 | [diff] [blame] | 21 | (strip_prefix, short_path)) |
| 22 | dest = ctx.actions.declare_file(short_path[len(strip_prefix):]) |
David L. Jones | b3cbea1 | 2022-05-12 19:48:58 -0700 | [diff] [blame] | 23 | src_dests.append([src, dest]) |
| 24 | |
| 25 | if ctx.attr.is_windows: |
| 26 | bat_file = ctx.actions.declare_file(ctx.label.name + "_copy.bat") |
| 27 | ctx.actions.write( |
| 28 | output = bat_file, |
| 29 | content = "\r\n".join([ |
| 30 | '@copy /Y "{}" "{}" >NUL'.format( |
| 31 | src.path.replace("/", "\\"), |
| 32 | dest.path.replace("/", "\\"), |
| 33 | ) |
| 34 | for src, dest in src_dests |
| 35 | ]) + "\r\n", |
| 36 | ) |
| 37 | ctx.actions.run( |
| 38 | inputs = ctx.files.srcs, |
| 39 | tools = [bat_file], |
| 40 | outputs = [dest for src, dest in src_dests], |
| 41 | executable = "cmd.exe", |
| 42 | arguments = ["/C", bat_file.path.replace("/", "\\")], |
| 43 | mnemonic = "InternalCopyFile", |
| 44 | progress_message = "Copying files", |
| 45 | use_default_shell_env = True, |
| 46 | ) |
| 47 | |
| 48 | else: |
| 49 | sh_file = ctx.actions.declare_file(ctx.label.name + "_copy.sh") |
| 50 | ctx.actions.write( |
| 51 | output = sh_file, |
| 52 | content = "\n".join([ |
| 53 | 'cp -f "{}" "{}"'.format(src.path, dest.path) |
| 54 | for src, dest in src_dests |
| 55 | ]), |
| 56 | ) |
| 57 | ctx.actions.run( |
| 58 | inputs = ctx.files.srcs, |
| 59 | tools = [sh_file], |
| 60 | outputs = [dest for src, dest in src_dests], |
| 61 | executable = "bash", |
| 62 | arguments = [sh_file.path], |
| 63 | mnemonic = "InternalCopyFile", |
| 64 | progress_message = "Copying files", |
| 65 | use_default_shell_env = True, |
| 66 | ) |
| 67 | |
| 68 | return [ |
| 69 | DefaultInfo(files = depset([dest for src, dest in src_dests])), |
| 70 | ] |
| 71 | |
| 72 | internal_copy_files_impl = rule( |
| 73 | doc = """ |
| 74 | Implementation for internal_copy_files macro. |
| 75 | |
| 76 | This rule implements file copying, including a compatibility mode for Windows. |
| 77 | """, |
| 78 | implementation = _internal_copy_files_impl, |
| 79 | attrs = { |
| 80 | "srcs": attr.label_list(allow_files = True, providers = [DefaultInfo]), |
| 81 | "strip_prefix": attr.string(), |
| 82 | "is_windows": attr.bool(), |
| 83 | }, |
| 84 | ) |
| 85 | |
David L. Jones | 07303d6 | 2022-05-17 18:13:22 -0700 | [diff] [blame] | 86 | def internal_copy_files(name, srcs, strip_prefix, **kwargs): |
David L. Jones | b3cbea1 | 2022-05-12 19:48:58 -0700 | [diff] [blame] | 87 | """Copies common proto files to the python tree. |
| 88 | |
| 89 | In order for Python imports to work, generated proto interfaces under |
| 90 | the google.protobuf package need to be in the same directory as other |
| 91 | source files. This rule copies the .proto files themselves, e.g. with |
| 92 | strip_prefix = 'src', 'src/google/protobuf/blah.proto' could be copied |
| 93 | to '<package>/google/protobuf/blah.proto'. |
| 94 | |
| 95 | (An alternative might be to implement a separate rule to generate |
| 96 | Python code in a different location for the sources. However, this |
| 97 | would be strange behavior that doesn't match any other language's proto |
| 98 | library generation.) |
| 99 | |
| 100 | Args: |
| 101 | name: the name for the rule. |
| 102 | srcs: the sources. |
| 103 | strip_prefix: the prefix to remove from each of the paths in 'srcs'. The |
| 104 | remainder will be used to construct the output path. |
David L. Jones | 07303d6 | 2022-05-17 18:13:22 -0700 | [diff] [blame] | 105 | **kwargs: common rule arguments. |
David L. Jones | b3cbea1 | 2022-05-12 19:48:58 -0700 | [diff] [blame] | 106 | |
| 107 | """ |
| 108 | internal_copy_files_impl( |
| 109 | name = name, |
| 110 | srcs = srcs, |
| 111 | strip_prefix = strip_prefix, |
| 112 | is_windows = select({ |
| 113 | "@bazel_tools//src/conditions:host_windows": True, |
| 114 | "//conditions:default": False, |
| 115 | }), |
David L. Jones | 07303d6 | 2022-05-17 18:13:22 -0700 | [diff] [blame] | 116 | **kwargs |
David L. Jones | b3cbea1 | 2022-05-12 19:48:58 -0700 | [diff] [blame] | 117 | ) |
Mike Kruskal | 3bc5049 | 2022-12-01 13:34:12 -0800 | [diff] [blame] | 118 | |
| 119 | def internal_py_test(deps = [], **kwargs): |
| 120 | """Internal wrapper for shared test configuration |
| 121 | |
| 122 | Args: |
| 123 | deps: any additional dependencies of the test. |
| 124 | **kwargs: arguments forwarded to py_test. |
| 125 | """ |
| 126 | native.py_test( |
| 127 | imports = ["."], |
Deanna Garcia | fe038fc | 2023-04-17 12:11:08 -0700 | [diff] [blame] | 128 | deps = deps + ["//python:python_test_lib"], |
Mike Kruskal | 3bc5049 | 2022-12-01 13:34:12 -0800 | [diff] [blame] | 129 | target_compatible_with = select({ |
| 130 | "@system_python//:supported": [], |
| 131 | "//conditions:default": ["@platforms//:incompatible"], |
| 132 | }), |
| 133 | **kwargs |
| 134 | ) |