Skip to content

Commit bb3aa9e

Browse files
committed
Swift: add --force to codegen
1 parent d6aad13 commit bb3aa9e

File tree

8 files changed

+92
-15
lines changed

8 files changed

+92
-15
lines changed

swift/codegen/codegen.py

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,8 @@ def _parse_args() -> argparse.Namespace:
4242
help="output directory for generated C++ files, required if trap or cpp is provided to --generate")
4343
p.add_argument("--generated-registry", type=_abspath, default=paths.swift_dir / "ql/.generated.list",
4444
help="registry file containing information about checked-in generated code")
45+
p.add_argument("--force", "-f", action="store_true",
46+
help="generate all files without skipping unchanged files and overwriting modified ones")
4547
return p.parse_args()
4648

4749

swift/codegen/generators/qlgen.py

Lines changed: 2 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -302,7 +302,8 @@ def generate(opts, renderer):
302302

303303
imports = {}
304304

305-
with renderer.manage(generated=generated, stubs=stubs, registry=opts.generated_registry) as renderer:
305+
with renderer.manage(generated=generated, stubs=stubs, registry=opts.generated_registry,
306+
force=opts.force) as renderer:
306307

307308
db_classes = [cls for cls in classes.values() if not cls.ipa]
308309
renderer.render(ql.DbClasses(db_classes), out / "Raw.qll")

swift/codegen/lib/render.py

Lines changed: 16 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -59,8 +59,8 @@ def _do_write(self, mnemonic: str, contents: str, output: pathlib.Path):
5959
log.debug(f"{mnemonic}: generated {output.name}")
6060

6161
def manage(self, generated: typing.Iterable[pathlib.Path], stubs: typing.Iterable[pathlib.Path],
62-
registry: pathlib.Path) -> "RenderManager":
63-
return RenderManager(self._swift_dir, generated, stubs, registry)
62+
registry: pathlib.Path, force: bool = False) -> "RenderManager":
63+
return RenderManager(self._swift_dir, generated, stubs, registry, force)
6464

6565

6666
class RenderManager(Renderer):
@@ -87,9 +87,10 @@ class Hashes:
8787

8888
def __init__(self, swift_dir: pathlib.Path, generated: typing.Iterable[pathlib.Path],
8989
stubs: typing.Iterable[pathlib.Path],
90-
registry: pathlib.Path):
90+
registry: pathlib.Path, force: bool = False):
9191
super().__init__(swift_dir)
9292
self._registry_path = registry
93+
self._force = force
9394
self._hashes = {}
9495
self.written = set()
9596
self._existing = set()
@@ -132,10 +133,13 @@ def _process_generated(self, generated: typing.Iterable[pathlib.Path]):
132133
for f in generated:
133134
self._existing.add(f)
134135
rel_path = self._get_path(f)
135-
if rel_path not in self._hashes:
136+
if self._force:
137+
pass
138+
elif rel_path not in self._hashes:
136139
log.warning(f"{rel_path} marked as generated but absent from the registry")
137140
elif self._hashes[rel_path].post != self._hash_file(f):
138-
raise Error(f"{rel_path} is generated but was modified, please revert the file")
141+
raise Error(f"{rel_path} is generated but was modified, please revert the file "
142+
"or pass --force to overwrite")
139143

140144
def _process_stubs(self, stubs: typing.Iterable[pathlib.Path]):
141145
for f in stubs:
@@ -144,10 +148,13 @@ def _process_stubs(self, stubs: typing.Iterable[pathlib.Path]):
144148
self._hashes.pop(rel_path, None)
145149
continue
146150
self._existing.add(f)
147-
if rel_path not in self._hashes:
151+
if self._force:
152+
pass
153+
elif rel_path not in self._hashes:
148154
log.warning(f"{rel_path} marked as stub but absent from the registry")
149155
elif self._hashes[rel_path].post != self._hash_file(f):
150-
raise Error(f"{rel_path} is a stub marked as generated, but it was modified")
156+
raise Error(f"{rel_path} is a stub marked as generated, but it was modified, "
157+
"please remove the `// generated` header, revert the file or pass --force to overwrite it")
151158

152159
@staticmethod
153160
def is_customized_stub(file: pathlib.Path) -> bool:
@@ -171,6 +178,8 @@ def _hash_string(data: str) -> str:
171178
return h.hexdigest()
172179

173180
def _load_registry(self):
181+
if self._force:
182+
return
174183
try:
175184
with open(self._registry_path) as reg:
176185
for line in reg:

swift/codegen/test/test_qlgen.py

Lines changed: 5 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,7 @@ def qlgen_opts(opts):
4848
opts.generated_registry = generated_registry_path()
4949
opts.ql_format = True
5050
opts.swift_dir = paths.swift_dir
51+
opts.force = False
5152
return opts
5253

5354

@@ -430,7 +431,9 @@ def test_format_error(opts, generate, render_manager, run_mock):
430431
generate([schema.Class('A')])
431432

432433

433-
def test_manage_parameters(opts, generate, renderer):
434+
@pytest.mark.parametrize("force", [False, True])
435+
def test_manage_parameters(opts, generate, renderer, force):
436+
opts.force = force
434437
ql_a = opts.ql_output / "A.qll"
435438
ql_b = opts.ql_output / "B.qll"
436439
stub_a = opts.ql_stub_output / "A.qll"
@@ -448,7 +451,7 @@ def test_manage_parameters(opts, generate, renderer):
448451
generate([schema.Class('A')])
449452
assert renderer.mock_calls == [
450453
mock.call.manage(generated={ql_a, ql_b, test_a, test_b, import_file()}, stubs={stub_a, stub_b},
451-
registry=opts.generated_registry)
454+
registry=opts.generated_registry, force=force)
452455
]
453456

454457

swift/codegen/test/test_render.py

Lines changed: 66 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,7 @@ def test_managed_render(pystache_renderer, sut):
7575
mock.call.render_name(data.template, data, generator=paths.exe_file.relative_to(paths.swift_dir)),
7676
]
7777

78+
7879
def test_managed_render_with_no_registry(pystache_renderer, sut):
7980
data = mock.Mock(spec=("template",))
8081
text = "some text"
@@ -266,5 +267,70 @@ def test_render_with_extensions(pystache_renderer, sut):
266267
assert_file(expected_output, expected_contents)
267268

268269

270+
def test_managed_render_with_force_not_skipping_generated_file(pystache_renderer, sut):
271+
data = mock.Mock(spec=("template",))
272+
output = paths.swift_dir / "some/output.txt"
273+
some_output = "some output"
274+
registry = paths.swift_dir / "a/registry.list"
275+
write(output, some_output)
276+
write(registry, f"some/output.txt {hash(some_output)} {hash(some_output)}\n")
277+
278+
pystache_renderer.render_name.side_effect = (some_output,)
279+
280+
with sut.manage(generated=(output,), stubs=(), registry=registry, force=True) as renderer:
281+
renderer.render(data, output)
282+
assert renderer.written == {output}
283+
assert_file(output, some_output)
284+
285+
assert_file(registry, f"some/output.txt {hash(some_output)} {hash(some_output)}\n")
286+
assert pystache_renderer.mock_calls == [
287+
mock.call.render_name(data.template, data, generator=paths.exe_file.relative_to(paths.swift_dir)),
288+
]
289+
290+
291+
def test_managed_render_with_force_not_skipping_stub_file(pystache_renderer, sut):
292+
data = mock.Mock(spec=("template",))
293+
stub = paths.swift_dir / "some/stub.txt"
294+
some_output = "// generated some output"
295+
some_processed_output = "// generated some processed output"
296+
registry = paths.swift_dir / "a/registry.list"
297+
write(stub, some_processed_output)
298+
write(registry, f"some/stub.txt {hash(some_output)} {hash(some_processed_output)}\n")
299+
300+
pystache_renderer.render_name.side_effect = (some_output,)
301+
302+
with sut.manage(generated=(), stubs=(stub,), registry=registry, force=True) as renderer:
303+
renderer.render(data, stub)
304+
assert renderer.written == {stub}
305+
assert_file(stub, some_output)
306+
307+
assert_file(registry, f"some/stub.txt {hash(some_output)} {hash(some_output)}\n")
308+
assert pystache_renderer.mock_calls == [
309+
mock.call.render_name(data.template, data, generator=paths.exe_file.relative_to(paths.swift_dir)),
310+
]
311+
312+
313+
def test_managed_render_with_force_ignores_modified_generated_file(sut):
314+
output = paths.swift_dir / "some/output.txt"
315+
some_processed_output = "// some processed output"
316+
registry = paths.swift_dir / "a/registry.list"
317+
write(output, "// something else")
318+
write(registry, f"some/output.txt whatever {hash(some_processed_output)}\n")
319+
320+
with sut.manage(generated=(output,), stubs=(), registry=registry, force=True):
321+
pass
322+
323+
324+
def test_managed_render_with_force_ignores_modified_stub_file_still_marked_as_generated(sut):
325+
stub = paths.swift_dir / "some/stub.txt"
326+
some_processed_output = "// generated some processed output"
327+
registry = paths.swift_dir / "a/registry.list"
328+
write(stub, "// generated something else")
329+
write(registry, f"some/stub.txt whatever {hash(some_processed_output)}\n")
330+
331+
with sut.manage(generated=(), stubs=(stub,), registry=registry, force=True):
332+
pass
333+
334+
269335
if __name__ == '__main__':
270336
sys.exit(pytest.main([__file__] + sys.argv[1:]))

swift/ql/.generated.list

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -385,7 +385,7 @@ ql/lib/codeql/swift/generated/Synth.qll 90df85be365c89c3c2e22041ee7dc9dd2ad9194b
385385
ql/lib/codeql/swift/generated/SynthConstructors.qll 5c91f09bd82728651ed61f498704e0f62847788fa986dec5e674d81f294076c7 5c91f09bd82728651ed61f498704e0f62847788fa986dec5e674d81f294076c7
386386
ql/lib/codeql/swift/generated/UnknownFile.qll 0fcf9beb8de79440bcdfff4bb6ab3dd139bd273e6c32754e05e6a632651e85f6 0fcf9beb8de79440bcdfff4bb6ab3dd139bd273e6c32754e05e6a632651e85f6
387387
ql/lib/codeql/swift/generated/UnknownLocation.qll e50efefa02a0ec1ff635a00951b5924602fc8cab57e5756e4a039382c69d3882 e50efefa02a0ec1ff635a00951b5924602fc8cab57e5756e4a039382c69d3882
388-
ql/lib/codeql/swift/generated/UnspecifiedElement.qll a3a73f53c492adc6655fb88b40b1bd9b2c9d365fc5c019b3233c01d6110fb3f2 ba77cd5272cffd1d3aad8bea69786b97aec0c93a4b59a070d621fe2d21c2e90c
388+
ql/lib/codeql/swift/generated/UnspecifiedElement.qll dbc6ca4018012977b26ca184a88044c55b0661e3998cd14d46295b62a8d69625 184c9a0ce18c2ac881943b0fb400613d1401ed1d5564f90716b6c310ba5afe71
389389
ql/lib/codeql/swift/generated/decl/AbstractFunctionDecl.qll 8255b24dddda83e8a7dee9d69a4cf9883b5a7ae43676d7242b5aab5169f68982 407c7d63681fb03ad6cb4ea3c2b04be7ccb5ddbe655a8aec4219eb3799bc36e8
390390
ql/lib/codeql/swift/generated/decl/AbstractStorageDecl.qll 66147ad36cefce974b4ae0f3e84569bd6742ea2f3e842c3c04e6e5cbd17e7928 ce7c2347e2dfe0b141db103ccb8e56a61d286476c201aebe6a275edd7fca2c0f
391391
ql/lib/codeql/swift/generated/decl/AbstractTypeParamDecl.qll 1e268b00d0f2dbbd85aa70ac206c5e4a4612f06ba0091e5253483635f486ccf9 5479e13e99f68f1f347283535f8098964f7fd4a34326ff36ad5711b2de1ab0d0

swift/ql/lib/codeql/swift/generated/UnspecifiedElement.qll

Lines changed: 0 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,6 @@ import codeql.swift.elements.Element
55
import codeql.swift.elements.ErrorElement
66

77
module Generated {
8-
/**
9-
* bla
10-
*/
118
class UnspecifiedElement extends Synth::TUnspecifiedElement, ErrorElement {
129
override string getAPrimaryQlClass() { result = "UnspecifiedElement" }
1310

swift/schema.py

Lines changed: 0 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,6 @@ class ErrorElement(Locatable):
4040

4141
@use_for_null
4242
class UnspecifiedElement(ErrorElement):
43-
"""bla"""
4443
parent: optional[Element]
4544
property: string
4645
index: optional[int]

0 commit comments

Comments
 (0)