Skip to content

Commit

Permalink
Fix #3290: Add support for generic regex pattern matching (#3340)
Browse files Browse the repository at this point in the history
* Fix #3290: Add support for generic regex pattern matching

* Apply review suggestions for proto location

* Delete main directory

* Implment script logic and tests

* CI setup part 1

* syntax fix in yaml file

* import nits in dummy file

* Use regex patterns from script in script test

* Make PR suggestions into effect

* Make PR suggestions into effect

* Make PR suggestions into effect

* Improve naming of script_assets variables

* Make PR suggestions into effect

* Make PR suggestions

* Make PR suggestions

* Revamp testing approach

* nit fixes

* Make nit suggestions

* Apply review suggestions on PR

* Refactor PR as per feedback recieved

* Implement review suggestions

* nits

* Nit fixes

* nit fix

* Review suggestions part 1

* Implement review suggestions part 2

* nit fixes

* update static_checks

* add test to repository file

* nit fix

* nit fix

* nit fix

* Implement review suggestions

* bazel files nit fix

* Introduce a library for the regex assets

* Make directory structure same as that of #3374

* diable ktlint max-line-length

* disable ktlint-max-line

* disable ktlint max-length

* nit fix

* nit fix

* Apply review suggestions

* ktlint fix

* nit fix

* Remove script constants

* nit fix

* add todo

* nit fixes

* add testOnly
  • Loading branch information
Sparsh1212 authored Jul 8, 2021
1 parent d99fd17 commit 72bd749
Show file tree
Hide file tree
Showing 19 changed files with 988 additions and 161 deletions.
3 changes: 3 additions & 0 deletions .github/CODEOWNERS
Original file line number Diff line number Diff line change
Expand Up @@ -225,3 +225,6 @@ WORKSPACE @BenHenning
*.bazel @BenHenning
.bazelrc @BenHenning
/tools/android/ @BenHenning

# Static analysis check configuration files ownership.
/scripts/assets/ @BenHenning
16 changes: 16 additions & 0 deletions .github/workflows/static_checks.yml
Original file line number Diff line number Diff line change
Expand Up @@ -92,3 +92,19 @@ jobs:
- name: Bazel lint check
run: |
bash /home/runner/work/oppia-android/oppia-android/scripts/buildifier_lint_check.sh $HOME
script_checks:
name: Script Checks
runs-on: ubuntu-18.04
steps:
- uses: actions/checkout@v2

- name: Set up Bazel
uses: abhinavsingh/setup-bazel@v3
with:
version: 4.0.0

- name: Regex Patterns Validation Check
if: always()
run: |
bazel run //scripts:regex_pattern_validation_check -- $(pwd)
191 changes: 31 additions & 160 deletions domain/domain_assets.bzl
Original file line number Diff line number Diff line change
Expand Up @@ -2,160 +2,7 @@
Macros for preparing & creating assets to include in the domain module.
"""

load("@rules_proto//proto:defs.bzl", "ProtoInfo")

def _extract_proto_sources(deps):
"""
Returns the list of proto source files that make up the specified list of proto dependencies.
The returned list includes transitive dependencies.
"""

# See https://github.com/bazelbuild/rules_proto/pull/77/files &
# https://github.com/bazelbuild/rules_proto/issues/57 &
# https://docs.bazel.build/versions/master/skylark/lib/ProtoInfo.html for references.
combined_sources = []
for dep in deps:
combined_sources.extend(dep[ProtoInfo].transitive_sources.to_list())
return combined_sources

def _gen_binary_proto_from_text_impl(ctx):
# See: https://docs.bazel.build/versions/master/skylark/lib/actions.html#declare_file.
output_file = ctx.outputs.output_file
input_file = ctx.attr.input_file.files.to_list()[0].short_path
input_proto_files = _extract_proto_sources(ctx.attr.proto_deps)

# See 'protoc --help' for specifics on the arguments passed to the tool for converting text
# proto to binary, and expected stdin/stdout configurations. Note that the actual proto files
# are passed to the compiler since it requires them in order to transcode the text proto file.
command_path = ctx.executable._protoc_tool.path
arguments = [command_path] + [
"--encode %s" % ctx.attr.proto_type_name,
] + [file.path for file in input_proto_files] + [
"< %s" % input_file,
"> %s" % output_file.path,
]

# Reference: https://docs.bazel.build/versions/master/skylark/lib/actions.html#run. This
# actually runs the proto compiler to perform the conversion. Note that this needs to use
# run_shell() instead of run() because it requires input redirection.
ctx.actions.run_shell(
outputs = [output_file],
inputs = ctx.files.input_file + input_proto_files,
tools = [ctx.executable._protoc_tool],
command = " ".join(arguments),
mnemonic = "GenerateBinaryProtoFromText",
)
return DefaultInfo(
files = depset([output_file]),
runfiles = ctx.runfiles(files = [output_file]),
)

# Custom Starlark rule for running the proto compiler in encode mode to convert a text proto to
# binary. The custom rule allows this to be done as part of the build graph so that binary files
# never need to be checked into the repository.
_gen_binary_proto_from_text = rule(
attrs = {
"input_file": attr.label(
allow_files = True,
mandatory = True,
),
"output_file": attr.output(
mandatory = True,
),
"proto_deps": attr.label_list(
allow_empty = False,
mandatory = True,
),
"proto_type_name": attr.string(mandatory = True),
"_protoc_tool": attr.label(
# This was partly inspired by https://stackoverflow.com/a/39138074.
executable = True,
cfg = "host",
default = "@protobuf_tools//:protoc",
),
},
implementation = _gen_binary_proto_from_text_impl,
)

def gen_binary_proto_from_text(name, proto_type_name, input_file, output_file, proto_deps):
"""
Generates a binary proto from a text proto.
Args:
name: str. A unique name to identify this generation. This can be built directly using Bazel
like any other build rule.
proto_type_name: str. The qualified type name of the proto being converted (e.g.
'model.Exploration').
input_file: file. The path to the text proto file being converted.
output_file: file. The output path for the generated binary proto file.
proto_deps: list of targets. The list of proto_library dependencies that are needed to
perform the conversion. Generally, only the proto file corresponding to the proto type
is needed since proto_library automatically pulls in transitive dependencies.
Returns:
str. The path to the newly generated binary file (same as output_file).
"""
_gen_binary_proto_from_text(
name = name,
proto_type_name = proto_type_name,
input_file = input_file,
output_file = output_file,
proto_deps = proto_deps,
)
return output_file

def _generate_single_asset_proto_binary(name, proto_file_name, proto_dep_name, proto_type_name):
"""
Converts a single asset text proto to a new binary asset.
Args:
name: str. The name of this target.
proto_file_name: str. The file name of the text proto under the assets directory that will
be converted. This is assuming to correspond to 'src/main/assets/<name>.textproto' and
will lead to a new generated file called 'src/main/assets/<name>.pb'.
proto_dep_name: str. The name of the proto library under //model that contains the proto
definition being converted to binary.
proto_type_name: str. The name of the proto type being converted in the text proto. This is
assumed to be part of the shared 'model' package.
Returns:
str. The path to the newly generated binary file.
"""
asset_dir = "src/main/assets"
return gen_binary_proto_from_text(
name = "generate_binary_proto_for_text_proto_%s" % name,
input_file = "%s/%s.textproto" % (asset_dir, proto_file_name),
output_file = "%s/%s.pb" % (asset_dir, proto_file_name),
proto_deps = [
"//model:%s_proto" % proto_dep_name,
],
proto_type_name = "model.%s" % proto_type_name,
)

def _generate_proto_binary_assets(names, proto_dep_name, proto_type_name, name_prefix):
"""
Converts a list of text proto assets to binary.
Args:
names: list of str. The list of text proto file names under the assets directory that should
be converted.
proto_dep_name: str. See _generate_single_asset_proto_binary.
proto_type_name: str. See _generate_single_asset_proto_binary.
name_prefix: str. A prefix to attach to the name of this target.
Returns:
list of str. The list of new proto binary asset files that were generated.
"""
return [
_generate_single_asset_proto_binary(
name = "%s_%s" % (name_prefix, name),
proto_file_name = name,
proto_dep_name = proto_dep_name,
proto_type_name = proto_type_name,
)
for name in names
]
load("//model:text_proto_assets.bzl", "generate_proto_binary_assets")

def generate_assets_list_from_text_protos(
name,
Expand All @@ -180,34 +27,58 @@ def generate_assets_list_from_text_protos(
Returns:
list of str. The list of new proto binary asset files that were generated.
"""
return _generate_proto_binary_assets(
return generate_proto_binary_assets(
name = name,
names = topic_list_file_names,
proto_dep_name = "topic",
proto_type_name = "TopicIdList",
name_prefix = name,
) + _generate_proto_binary_assets(
asset_dir = "src/main/assets",
proto_dep_bazel_target_prefix = "//model",
proto_package = "model",
) + generate_proto_binary_assets(
name = name,
names = topic_file_names,
proto_dep_name = "topic",
proto_type_name = "TopicRecord",
name_prefix = name,
) + _generate_proto_binary_assets(
asset_dir = "src/main/assets",
proto_dep_bazel_target_prefix = "//model",
proto_package = "model",
) + generate_proto_binary_assets(
name = name,
names = subtopic_file_names,
proto_dep_name = "topic",
proto_type_name = "SubtopicRecord",
name_prefix = name,
) + _generate_proto_binary_assets(
asset_dir = "src/main/assets",
proto_dep_bazel_target_prefix = "//model",
proto_package = "model",
) + generate_proto_binary_assets(
name = name,
names = story_file_names,
proto_dep_name = "topic",
proto_type_name = "StoryRecord",
name_prefix = name,
) + _generate_proto_binary_assets(
asset_dir = "src/main/assets",
proto_dep_bazel_target_prefix = "//model",
proto_package = "model",
) + generate_proto_binary_assets(
name = name,
names = skills_file_names,
proto_dep_name = "topic",
proto_type_name = "ConceptCardList",
name_prefix = name,
) + _generate_proto_binary_assets(
asset_dir = "src/main/assets",
proto_dep_bazel_target_prefix = "//model",
proto_package = "model",
) + generate_proto_binary_assets(
name = name,
names = exploration_file_names,
proto_dep_name = "exploration",
proto_type_name = "Exploration",
name_prefix = name,
asset_dir = "src/main/assets",
proto_dep_bazel_target_prefix = "//model",
proto_package = "model",
)
Loading

2 comments on commit 72bd749

@ayeshasiddika9196

This comment was marked as spam.

@ayeshasiddika9196

This comment was marked as spam.

Please sign in to comment.