From 7098a0d3963c0bd67e7b65a5a601621d11d25e3b Mon Sep 17 00:00:00 2001 From: Chad Woolley <cwoolley@gitlab.com> Date: Tue, 30 Aug 2022 18:46:19 +0000 Subject: [PATCH] DRY up and clean up GLFM fixtures - Extract common fixture setup to new shared context - Move shared contexts to common glfm directory - Renaming files for consistency - Clean up and improve some file reading and other logic MR: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/96325 --- .../specification_guide/index.md | 79 ++++- .../requests/api/markdown_snapshot_spec.rb | 10 + .../example_snapshots/examples_index.yml | 18 ++ glfm_specification/example_snapshots/html.yml | 42 +++ .../example_snapshots/markdown.yml | 12 + .../example_snapshots/prosemirror_json.yml | 12 + .../glfm_canonical_examples.txt | 61 ++++ .../glfm_example_metadata.yml | 14 + .../glfm_example_status.yml | 36 +++ glfm_specification/output/spec.txt | 61 ++++ scripts/glfm/run-snapshot-tests.sh | 6 +- scripts/lib/glfm/constants.rb | 8 +- scripts/lib/glfm/render_static_html.rb | 27 +- scripts/lib/glfm/update_example_snapshots.rb | 54 ++-- spec/requests/api/markdown_snapshot_spec.rb | 4 +- spec/scripts/lib/glfm/parse_examples_spec.rb | 8 +- .../lib/glfm/update_example_snapshots_spec.rb | 306 ++++++++++++++++-- .../lib/glfm/update_specification_spec.rb | 21 +- .../api_markdown_snapshot_shared_context.rb} | 34 +- .../glfm/example_snapshot_fixtures.rb | 27 ++ 20 files changed, 723 insertions(+), 117 deletions(-) create mode 100644 ee/spec/requests/api/markdown_snapshot_spec.rb create mode 100644 glfm_specification/input/gitlab_flavored_markdown/glfm_example_metadata.yml rename spec/support/shared_contexts/{markdown_snapshot_shared_examples.rb => glfm/api_markdown_snapshot_shared_context.rb} (66%) create mode 100644 spec/support/shared_contexts/glfm/example_snapshot_fixtures.rb diff --git a/doc/development/gitlab_flavored_markdown/specification_guide/index.md b/doc/development/gitlab_flavored_markdown/specification_guide/index.md index f2d813cb993f8..c1227e5d33f4c 100644 --- a/doc/development/gitlab_flavored_markdown/specification_guide/index.md +++ b/doc/development/gitlab_flavored_markdown/specification_guide/index.md @@ -345,8 +345,50 @@ For the [Markdown snapshot testing](#markdown-snapshot-testing) to work properly, you must account for these differences in a way that ensures the tests are reliable, and always behave the same across different test runs or environments. -To account for these differences, there is a process called **_normalization_**. Normalization -allows custom regular expressions with +To account for these differences, there is a process called **_normalization_**. Several ways to approach normalization exist: + +1. Fixture-based normalization +1. Environment-variable-based normalization +1. Regex-based normalization + +#### Fixture-based normalization + +Fixture-based normalization should be used whenever possible, because it is simpler and easier to +understand than regex-based normalization. + +The [Markdown snapshot testing](#markdown-snapshot-testing) uses RSpec to generate the +[example snapshot files](#example-snapshot-files). RSpec enables you to: + +- Use the same powerful fixture support and helpers as all the rest of the GitLab RSpec suite. +- Use fixtures to control the state of the database when the example snapshots are generated. +- Extract this fixture setup to an RSpec shared context. This shared context is used to ensure + the same database state exists wherever the snapshot tests are run, either by the CI suite, or + locally via [`run-snapshot-tests.sh`](#run-snapshot-testssh-script). + +You can see the RSpec shared context containing these fixtures in +[`spec/support/shared_contexts/glfm/example_snapshot_fixtures.rb`](https://gitlab.com/gitlab-org/gitlab/blob/master/spec/support/shared_contexts/glfm/example_snapshot_fixtures.rb). + +#### Environment-variable-based normalization + +In some cases, fixtures may not be usable, because they do not provide control over the varying +values. In these cases, we can introduce support for a environment variable into the production +code, which allows us to override the randommness in our test environment when we are +generating the HTML for footnote examples. Even though it is in the production code path, it has +no effect unless it is explicitly set, therefore it is innocuous. It allows us to avoid +the more-complex regex-based normalization described below. + +The current example of this is when normally random footnote IDs are overridden to be deterministic +by setting `GITLAB_TEST_FOOTNOTE_ID`. It is set along with the fixtures setup in the +[`spec/support/shared_contexts/glfm/example_snapshot_fixtures.rb`](https://gitlab.com/gitlab-org/gitlab/blob/master/spec/support/shared_contexts/glfm/example_snapshot_fixtures.rb) +shared context. + +#### Regex-based normalization + +If neither fixture-based nor environment-variable-based normalization can be used, use regex-based +normalization. It is powerful, but more complex, and requires more maintenance. +It requires referring to specific examples by name, and crafting the proper regexes. + +Regex-based normalization allows custom regular expressions with [_capturing groups_](https://ruby-doc.org/core-3.1.2/Regexp.html#class-Regexp-label-Capturing) to be applied to two different versions of HTML or JSON for a given Markdown example, and the contents of the captured groups can be replaced with the same fixed values. @@ -813,6 +855,39 @@ to be specified for a Markdown example. 07_01_00_id: *07_01_00_id ``` +##### `glfm_example_metadata.yml` + +[`glfm_specification/input/gitlab_flavored_markdown/glfm_example_metadata.yml`](https://gitlab.com/gitlab-org/gitlab/-/blob/master/glfm_specification/input/gitlab_flavored_markdown/glfm_example_metadata.yml) +allows control over other aspects of the snapshot example generation process. + +- It is manually updated. +- The `ee` fields determine whether the example is an EE-only example. If the `ee` field is `true`, + the example will only be run by `ee/spec/requests/api/markdown_snapshot_spec.rb`, not by + `spec/requests/api/markdown_snapshot_spec.rb`. +- The `api_request_override_path` field overrides the API endpoint path which is used to + generate the `static` HTML for the specifed example. Different endpoints can generate different + HTML in some cases, so we want to be able to exercise different API endpoints for the same + Markdown. By default, the `/markdown` endpoint is used. + +`glfm_specification/input/gitlab_flavored_markdown/glfm_example_metadata.yml` sample entries: + +```yaml +--- +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__001: + api_request_override_path: /groups/glfm_group/preview_markdown +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__002: + api_request_override_path: /glfm_group/glfm_project/preview_markdown +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__003: + api_request_override_path: /glfm_group/glfm_project/preview_markdown +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__004: + api_request_override_path: /-/snippets/preview_markdown +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__005: + api_request_override_path: /glfm_group/glfm_project/-/wikis/new_page/preview_markdown +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__006: + ee: true + api_request_override_path: /groups/glfm_group/-/wikis/new_page/preview_markdown +``` + #### Output specification files The `glfm_specification/output` directory contains the CommonMark standard format diff --git a/ee/spec/requests/api/markdown_snapshot_spec.rb b/ee/spec/requests/api/markdown_snapshot_spec.rb new file mode 100644 index 0000000000000..66d3c60d6f619 --- /dev/null +++ b/ee/spec/requests/api/markdown_snapshot_spec.rb @@ -0,0 +1,10 @@ +# frozen_string_literal: true + +require 'spec_helper' + +# See https://docs.gitlab.com/ee/development/gitlab_flavored_markdown/specification_guide/#markdown-snapshot-testing +# for documentation on this spec. +RSpec.describe API::Markdown, 'Snapshot' do + # noinspection RailsParamDefResolve (RubyMine can't find the shared context from this file location) + include_context 'with API::Markdown Snapshot shared context', ee_only: true +end diff --git a/glfm_specification/example_snapshots/examples_index.yml b/glfm_specification/example_snapshots/examples_index.yml index bd295baaad0bb..3d4efcfbd7a90 100644 --- a/glfm_specification/example_snapshots/examples_index.yml +++ b/glfm_specification/example_snapshots/examples_index.yml @@ -2054,3 +2054,21 @@ 07_05_00__gitlab_specific_markdown__video__002: spec_txt_example_position: 687 source_specification: gitlab +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__001: + spec_txt_example_position: 688 + source_specification: gitlab +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__002: + spec_txt_example_position: 689 + source_specification: gitlab +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__003: + spec_txt_example_position: 690 + source_specification: gitlab +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__004: + spec_txt_example_position: 691 + source_specification: gitlab +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__005: + spec_txt_example_position: 692 + source_specification: gitlab +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__006: + spec_txt_example_position: 693 + source_specification: gitlab diff --git a/glfm_specification/example_snapshots/html.yml b/glfm_specification/example_snapshots/html.yml index 196cd4f748dbc..1df9354822a9b 100644 --- a/glfm_specification/example_snapshots/html.yml +++ b/glfm_specification/example_snapshots/html.yml @@ -7850,3 +7850,45 @@ wysiwyg: |- <pre>[video]: video.mov "video title"</pre> <p><span class="media-container video-container"><video src="video.mov" controls="true" data-setup="{}" data-title="video"></video><a href="video.mov">video</a></span></p> +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__001: + canonical: | + <p><a href="groups-test-file">groups-test-file</a></p> + static: |- + <p data-sourcepos="1:1-1:45" dir="auto"><a href="/groups/glfm_group/-/uploads/groups-test-file" data-canonical-src="/uploads/groups-test-file" data-link="true" class="gfm">groups-test-file</a></p> + wysiwyg: |- + Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__002: + canonical: | + <p><a href="projects-test-file">projects-test-file</a></p> + static: |- + <p data-sourcepos="1:1-1:40" dir="auto"><a href="/glfm_group/glfm_project/-/blob/master/projects-test-file">projects-test-file</a></p> + wysiwyg: |- + Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__003: + canonical: | + <p>This project snippet ID reference IS filtered: $88888</p> + static: |- + <p data-sourcepos="1:1-1:53" dir="auto">This project snippet ID reference IS filtered: <a href="/glfm_group/glfm_project/-/snippets/88888" data-reference-type="snippet" data-original="$88888" data-link="false" data-link-reference="false" data-project="77777" data-snippet="88888" data-container="body" data-placement="top" title="glfm_project_snippet" class="gfm gfm-snippet has-tooltip">$88888</a></p> + wysiwyg: |- + Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__004: + canonical: | + <p>This personal snippet ID reference is not filtered: $99999</p> + static: |- + <p data-sourcepos="1:1-1:58" dir="auto">This personal snippet ID reference is not filtered: $99999</p> + wysiwyg: |- + Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__005: + canonical: | + <p><a href="project-wikis-test-file">project-wikis-test-file</a></p> + static: |- + <p data-sourcepos="1:1-1:50" dir="auto"><a href="/glfm_group/glfm_project/-/wikis/project-wikis-test-file" data-canonical-src="project-wikis-test-file">project-wikis-test-file</a></p> + wysiwyg: |- + Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__006: + canonical: | + <p><a href="group-wikis-test-file">group-wikis-test-file</a></p> + static: |- + <p data-sourcepos="1:1-1:46" dir="auto"><a href="/groups/glfm_group/-/wikis/group-wikis-test-file" data-canonical-src="group-wikis-test-file">group-wikis-test-file</a></p> + wysiwyg: |- + Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 diff --git a/glfm_specification/example_snapshots/markdown.yml b/glfm_specification/example_snapshots/markdown.yml index c2b2caacb5bdb..e928c5a977e07 100644 --- a/glfm_specification/example_snapshots/markdown.yml +++ b/glfm_specification/example_snapshots/markdown.yml @@ -2239,3 +2239,15 @@ [video]: video.mov "video title" ![video][video] +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__001: | + [groups-test-file](/uploads/groups-test-file) +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__002: | + [projects-test-file](projects-test-file) +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__003: | + This project snippet ID reference IS filtered: $88888 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__004: | + This personal snippet ID reference is not filtered: $99999 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__005: | + [project-wikis-test-file](project-wikis-test-file) +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__006: | + [group-wikis-test-file](group-wikis-test-file) diff --git a/glfm_specification/example_snapshots/prosemirror_json.yml b/glfm_specification/example_snapshots/prosemirror_json.yml index 02330019a8d8a..7928d50ae961c 100644 --- a/glfm_specification/example_snapshots/prosemirror_json.yml +++ b/glfm_specification/example_snapshots/prosemirror_json.yml @@ -20890,3 +20890,15 @@ } ] } +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__001: |- + Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__002: |- + Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__003: |- + Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__004: |- + Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__005: |- + Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__006: |- + Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 diff --git a/glfm_specification/input/gitlab_flavored_markdown/glfm_canonical_examples.txt b/glfm_specification/input/gitlab_flavored_markdown/glfm_canonical_examples.txt index fd540a905bd67..d27f809aaf350 100644 --- a/glfm_specification/input/gitlab_flavored_markdown/glfm_canonical_examples.txt +++ b/glfm_specification/input/gitlab_flavored_markdown/glfm_canonical_examples.txt @@ -250,3 +250,64 @@ Reference definitions work video as well: . <p><video src="video.mov" title="video title"></video></p> ```````````````````````````````` + +# Examples Using Internal Extensions + +## Markdown Preview API Request Overrides + +This section contains examples of all controllers which use `PreviewMarkdown` module +and use different `markdown_context_params`. They exercise the various `preview_markdown` +endpoints via `glfm_example_metadata.yml`. + + +`preview_markdown` exercising `groups` API endpoint and `UploadLinkFilter`: + +```````````````````````````````` example gitlab +[groups-test-file](/uploads/groups-test-file) +. +<p><a href="groups-test-file">groups-test-file</a></p> +```````````````````````````````` + +`preview_markdown` exercising `projects` API endpoint and `RepositoryLinkFilter`: + +```````````````````````````````` example gitlab +[projects-test-file](projects-test-file) +. +<p><a href="projects-test-file">projects-test-file</a></p> +```````````````````````````````` + +`preview_markdown` exercising `projects` API endpoint and `SnippetReferenceFilter`: + +```````````````````````````````` example gitlab +This project snippet ID reference IS filtered: $88888 +. +<p>This project snippet ID reference IS filtered: $88888</p> +```````````````````````````````` + +`preview_markdown` exercising personal (non-project) `snippets` API endpoint. This is +only used by the comment field on personal snippets. It has no unique custom markdown +extension behavior, and specifically does not render snippet references via +`SnippetReferenceFilter`, even if the ID is valid. + +```````````````````````````````` example gitlab +This personal snippet ID reference is not filtered: $99999 +. +<p>This personal snippet ID reference is not filtered: $99999</p> +```````````````````````````````` + +`preview_markdown` exercising project `wikis` API endpoint and `WikiLinkFilter`: + +```````````````````````````````` example gitlab +[project-wikis-test-file](project-wikis-test-file) +. +<p><a href="project-wikis-test-file">project-wikis-test-file</a></p> +```````````````````````````````` + +`preview_markdown` exercising group `wikis` API endpoint and `WikiLinkFilter`. This example +also requires an EE license enabling the `group_wikis` feature: + +```````````````````````````````` example gitlab +[group-wikis-test-file](group-wikis-test-file) +. +<p><a href="group-wikis-test-file">group-wikis-test-file</a></p> +```````````````````````````````` diff --git a/glfm_specification/input/gitlab_flavored_markdown/glfm_example_metadata.yml b/glfm_specification/input/gitlab_flavored_markdown/glfm_example_metadata.yml new file mode 100644 index 0000000000000..3c043f5fba117 --- /dev/null +++ b/glfm_specification/input/gitlab_flavored_markdown/glfm_example_metadata.yml @@ -0,0 +1,14 @@ +--- +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__001: + api_request_override_path: /groups/glfm_group/preview_markdown +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__002: + api_request_override_path: /glfm_group/glfm_project/preview_markdown +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__003: + api_request_override_path: /glfm_group/glfm_project/preview_markdown +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__004: + api_request_override_path: /-/snippets/preview_markdown +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__005: + api_request_override_path: /glfm_group/glfm_project/-/wikis/new_page/preview_markdown +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__006: + ee: true + api_request_override_path: /groups/glfm_group/-/wikis/new_page/preview_markdown diff --git a/glfm_specification/input/gitlab_flavored_markdown/glfm_example_status.yml b/glfm_specification/input/gitlab_flavored_markdown/glfm_example_status.yml index 512f197047f8f..a74c3492324e9 100644 --- a/glfm_specification/input/gitlab_flavored_markdown/glfm_example_status.yml +++ b/glfm_specification/input/gitlab_flavored_markdown/glfm_example_status.yml @@ -24,3 +24,39 @@ skip_running_conformance_wysiwyg_tests: Inapplicable task list items not yet implemented for WYSYWIG skip_running_snapshot_wysiwyg_html_tests: Inapplicable task list items not yet implemented for WYSYWIG skip_running_snapshot_prosemirror_json_tests: Inapplicable task list items not yet implemented for WYSYWIG +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__001: + skip_update_example_snapshot_html_wysiwyg: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_update_example_snapshot_prosemirror_json: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_conformance_wysiwyg_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_snapshot_wysiwyg_html_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_snapshot_prosemirror_json_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__002: + skip_update_example_snapshot_html_wysiwyg: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_update_example_snapshot_prosemirror_json: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_conformance_wysiwyg_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_snapshot_wysiwyg_html_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_snapshot_prosemirror_json_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__003: + skip_update_example_snapshot_html_wysiwyg: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_update_example_snapshot_prosemirror_json: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_conformance_wysiwyg_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_snapshot_wysiwyg_html_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_snapshot_prosemirror_json_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__004: + skip_update_example_snapshot_html_wysiwyg: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_update_example_snapshot_prosemirror_json: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_conformance_wysiwyg_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_snapshot_wysiwyg_html_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_snapshot_prosemirror_json_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__005: + skip_update_example_snapshot_html_wysiwyg: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_update_example_snapshot_prosemirror_json: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_conformance_wysiwyg_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_snapshot_wysiwyg_html_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_snapshot_prosemirror_json_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 +08_01_00__examples_using_internal_extensions__markdown_preview_api_request_overrides__006: + skip_update_example_snapshot_html_wysiwyg: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_update_example_snapshot_prosemirror_json: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_conformance_wysiwyg_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_snapshot_wysiwyg_html_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 + skip_running_snapshot_prosemirror_json_tests: Not yet implemented. See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/92507#note_1068159236 diff --git a/glfm_specification/output/spec.txt b/glfm_specification/output/spec.txt index 2939e0d2fbde5..d79a8d6b6758a 100644 --- a/glfm_specification/output/spec.txt +++ b/glfm_specification/output/spec.txt @@ -9853,6 +9853,67 @@ Reference definitions work video as well: <p><video src="video.mov" title="video title"></video></p> ```````````````````````````````` +# Examples Using Internal Extensions + +## Markdown Preview API Request Overrides + +This section contains examples of all controllers which use `PreviewMarkdown` module +and use different `markdown_context_params`. They exercise the various `preview_markdown` +endpoints via `glfm_example_metadata.yml`. + + +`preview_markdown` exercising `groups` API endpoint and `UploadLinkFilter`: + +```````````````````````````````` example gitlab +[groups-test-file](/uploads/groups-test-file) +. +<p><a href="groups-test-file">groups-test-file</a></p> +```````````````````````````````` + +`preview_markdown` exercising `projects` API endpoint and `RepositoryLinkFilter`: + +```````````````````````````````` example gitlab +[projects-test-file](projects-test-file) +. +<p><a href="projects-test-file">projects-test-file</a></p> +```````````````````````````````` + +`preview_markdown` exercising `projects` API endpoint and `SnippetReferenceFilter`: + +```````````````````````````````` example gitlab +This project snippet ID reference IS filtered: $88888 +. +<p>This project snippet ID reference IS filtered: $88888</p> +```````````````````````````````` + +`preview_markdown` exercising personal (non-project) `snippets` API endpoint. This is +only used by the comment field on personal snippets. It has no unique custom markdown +extension behavior, and specifically does not render snippet references via +`SnippetReferenceFilter`, even if the ID is valid. + +```````````````````````````````` example gitlab +This personal snippet ID reference is not filtered: $99999 +. +<p>This personal snippet ID reference is not filtered: $99999</p> +```````````````````````````````` + +`preview_markdown` exercising project `wikis` API endpoint and `WikiLinkFilter`: + +```````````````````````````````` example gitlab +[project-wikis-test-file](project-wikis-test-file) +. +<p><a href="project-wikis-test-file">project-wikis-test-file</a></p> +```````````````````````````````` + +`preview_markdown` exercising group `wikis` API endpoint and `WikiLinkFilter`. This example +also requires an EE license enabling the `group_wikis` feature: + +```````````````````````````````` example gitlab +[group-wikis-test-file](group-wikis-test-file) +. +<p><a href="group-wikis-test-file">group-wikis-test-file</a></p> +```````````````````````````````` + <!-- END TESTS --> # Appendix: A parsing strategy diff --git a/scripts/glfm/run-snapshot-tests.sh b/scripts/glfm/run-snapshot-tests.sh index 59a7c8f06b079..6a66d8fbd9a6a 100755 --- a/scripts/glfm/run-snapshot-tests.sh +++ b/scripts/glfm/run-snapshot-tests.sh @@ -28,8 +28,12 @@ printf "\n${BBlue}Running frontend 'yarn jest spec/frontend/content_editor/markd yarn jest spec/frontend/content_editor/markdown_snapshot_spec.js printf "\n${BBlue}'yarn jest spec/frontend/content_editor/markdown_snapshot_spec.js' passed!${Color_Off}\n\n" -printf "\n${BBlue}Running backend 'bundle exec rspec spec/requests/api/markdown_snapshot_spec.rb'...${Color_Off}\n\n" +printf "\n${BBlue}Running CE backend 'bundle exec rspec spec/requests/api/markdown_snapshot_spec.rb'...${Color_Off}\n\n" bundle exec rspec spec/requests/api/markdown_snapshot_spec.rb printf "\n${BBlue}'bundle exec rspec spec/requests/api/markdown_snapshot_spec.rb' passed!${Color_Off}\n\n" +printf "\n${BBlue}Running EE backend 'bundle exec rspec ee/spec/requests/api/markdown_snapshot_spec.rb'...${Color_Off}\n\n" +bundle exec rspec ee/spec/requests/api/markdown_snapshot_spec.rb +printf "\n${BBlue}'bundle exec rspec ee/spec/requests/api/markdown_snapshot_spec.rb' passed!${Color_Off}\n\n" + printf "\n✅✅✅ ${BGreen}All GLFM snapshot example tests passed successfully!${Color_Off} ✅✅✅\n" diff --git a/scripts/lib/glfm/constants.rb b/scripts/lib/glfm/constants.rb index 42977248c0d7c..e5790bbdd8831 100644 --- a/scripts/lib/glfm/constants.rb +++ b/scripts/lib/glfm/constants.rb @@ -18,6 +18,9 @@ module Constants GLFM_INTRO_TXT_PATH = specification_input_glfm_path.join('glfm_intro.txt') GLFM_EXAMPLES_TXT_PATH = specification_input_glfm_path.join('glfm_canonical_examples.txt') GLFM_EXAMPLE_STATUS_YML_PATH = specification_input_glfm_path.join('glfm_example_status.yml') + GLFM_EXAMPLE_METADATA_YML_PATH = + specification_input_glfm_path.join('glfm_example_metadata.yml') + GLFM_EXAMPLE_NORMALIZATIONS_YML_PATH = specification_input_glfm_path.join('glfm_example_normalizations.yml') GLFM_SPEC_TXT_PATH = specification_path.join('output/spec.txt') # Example Snapshot (ES) files @@ -28,15 +31,16 @@ module Constants ES_PROSEMIRROR_JSON_YML_PATH = File.join(es_fixtures_path, 'prosemirror_json.yml') # Other constants used for processing files - GLFM_SPEC_TXT_HEADER = <<~GLFM_SPEC_TXT_HEADER + GLFM_SPEC_TXT_HEADER = <<~MARKDOWN --- title: GitLab Flavored Markdown (GLFM) Spec version: alpha ... - GLFM_SPEC_TXT_HEADER + MARKDOWN INTRODUCTION_HEADER_LINE_TEXT = /\A# Introduction\Z/.freeze END_TESTS_COMMENT_LINE_TEXT = /\A<!-- END TESTS -->\Z/.freeze MARKDOWN_TEMPFILE_BASENAME = %w[MARKDOWN_TEMPFILE_ .yml].freeze + METADATA_TEMPFILE_BASENAME = %w[METADATA_TEMPFILE_ .yml].freeze STATIC_HTML_TEMPFILE_BASENAME = %w[STATIC_HTML_TEMPFILE_ .yml].freeze WYSIWYG_HTML_AND_JSON_TEMPFILE_BASENAME = %w[WYSIWYG_HTML_AND_JSON_TEMPFILE_ .yml].freeze end diff --git a/scripts/lib/glfm/render_static_html.rb b/scripts/lib/glfm/render_static_html.rb index 25abc75e2709c..8d72aec7c3b88 100644 --- a/scripts/lib/glfm/render_static_html.rb +++ b/scripts/lib/glfm/render_static_html.rb @@ -29,34 +29,19 @@ include Glfm::Constants include Glfm::Shared - # TODO: Remove duplication of fixtures & logic with spec/support/shared_contexts/markdown_snapshot_shared_examples.rb - - let_it_be(:user) { create(:user) } - let_it_be(:group) { create(:group, name: 'glfm_group').tap { |group| group.add_owner(user) } } - - let_it_be(:project) do - # NOTE: We hardcode the IDs on all fixtures to prevent variability in the - # rendered HTML/Prosemirror JSON, and to minimize the need for normalization: - # https://docs.gitlab.com/ee/development/gitlab_flavored_markdown/specification_guide/#normalization - create(:project, :repository, creator: user, group: group, name: 'glfm_project', id: 77777) - end - - let_it_be(:project_snippet) { create(:project_snippet, id: 88888, project: project) } - let_it_be(:personal_snippet) { create(:snippet, id: 99999) } - - before do - stub_licensed_features(group_wikis: true) - sign_in(user) - end + # noinspection RailsParamDefResolve (RubyMine can't find the shared context from this file location) + include_context 'with GLFM example snapshot fixtures' it 'can create a project dependency graph using factories' do markdown_hash = YAML.safe_load(File.open(ENV.fetch('INPUT_MARKDOWN_YML_PATH')), symbolize_names: true) + metadata_hash = YAML.safe_load(File.open(ENV.fetch('INPUT_METADATA_YML_PATH')), symbolize_names: true) # NOTE: We cannot parallelize this loop like the Javascript WYSIWYG example generation does, # because the rspec `post` API cannot be parallized (it is not thread-safe, it can't find # the controller). - static_html_hash = markdown_hash.transform_values do |markdown| - api_url = api "/markdown" + static_html_hash = markdown_hash.transform_values.with_index do |markdown, index| + name = markdown_hash.keys[index] + api_url = metadata_hash.dig(name, :api_request_override_path) || (api "/markdown") post api_url, params: { text: markdown, gfm: true } # noinspection RubyResolve diff --git a/scripts/lib/glfm/update_example_snapshots.rb b/scripts/lib/glfm/update_example_snapshots.rb index 6ff063c169087..7dc0d0f7c4bc2 100644 --- a/scripts/lib/glfm/update_example_snapshots.rb +++ b/scripts/lib/glfm/update_example_snapshots.rb @@ -29,8 +29,6 @@ class UpdateExampleSnapshots def process(skip_static_and_wysiwyg: false) output('Updating example snapshots...') - setup_environment - output('(Skipping static HTML generation)') if skip_static_and_wysiwyg output("Reading #{GLFM_SPEC_TXT_PATH}...") @@ -49,14 +47,6 @@ def process(skip_static_and_wysiwyg: false) private - def setup_environment - # Set 'GITLAB_TEST_FOOTNOTE_ID' in order to override random number generation in - # Banzai::Filter::FootnoteFilter#random_number, and thus avoid the need to - # perform normalization on the value. See: - # https://docs.gitlab.com/ee/development/gitlab_flavored_markdown/specification_guide/#normalization - ENV['GITLAB_TEST_FOOTNOTE_ID'] = '42' - end - def add_example_names(all_examples) # NOTE: This method and the parse_examples method assume: # 1. Section 2 is the first section which contains examples @@ -137,11 +127,11 @@ def write_snapshot_example_files(all_examples, skip_static_and_wysiwyg:) return end - # NOTE: We pass the INPUT_MARKDOWN_YML_PATH via - # environment variable to the static/wysiwyg HTML generation scripts. This is because it - # is implemented as a subprocess which invokes rspec/jest scripts, and rspec/jest do not make + # NOTE: We pass the INPUT_MARKDOWN_YML_PATH and INPUT_METADATA_YML_PATH via + # environment variables to the static/wysiwyg HTML generation scripts. This is because they + # are implemented as subprocesses which invoke rspec/jest scripts, and rspec/jest do not make # it straightforward to pass arguments via the command line. - ENV['INPUT_MARKDOWN_YML_PATH'] = copy_tempfiles_for_subprocesses + ENV['INPUT_MARKDOWN_YML_PATH'], ENV['INPUT_METADATA_YML_PATH'] = copy_tempfiles_for_subprocesses static_html_hash = generate_static_html wysiwyg_html_and_json_hash = generate_wysiwyg_html_and_json @@ -168,18 +158,31 @@ def write_examples_index_yml(all_examples) name = example.fetch(:name).to_sym hash[name] = { 'spec_txt_example_position' => example.fetch(:example), - 'source_specification' => - if example[:extensions].empty? - 'commonmark' - elsif example[:extensions].include?('gitlab') - 'gitlab' - else - 'github' - end + 'source_specification' => source_specification_for_extensions(example.fetch(:extensions)) } end end + def source_specification_for_extensions(extensions) + unprocessed_extensions = extensions.map(&:to_sym) + unprocessed_extensions.delete(:disabled) + + source_specification = + if unprocessed_extensions.empty? + 'commonmark' + elsif unprocessed_extensions.include?(:gitlab) + unprocessed_extensions.delete(:gitlab) + 'gitlab' + else + 'github' + end + + # We should only be left with at most one extension, which is an optional name for the example + raise "Error: Invalid extension(s) found: #{unprocessed_extensions.join(', ')}" if unprocessed_extensions.size > 1 + + source_specification + end + def write_markdown_yml(all_examples) generate_and_write_for_all_examples(all_examples, ES_MARKDOWN_YML_PATH) do |example, hash| name = example.fetch(:name).to_sym @@ -193,14 +196,17 @@ def copy_tempfiles_for_subprocesses # the scripts to read them, because the scripts are run in # separate subprocesses, and during unit testing we are unable to substitute the mock # StringIO when reading the input files in the subprocess. - { ES_MARKDOWN_YML_PATH => MARKDOWN_TEMPFILE_BASENAME }.map do |original_file_path, tempfile_basename| + { + ES_MARKDOWN_YML_PATH => MARKDOWN_TEMPFILE_BASENAME, + GLFM_EXAMPLE_METADATA_YML_PATH => METADATA_TEMPFILE_BASENAME + }.map do |original_file_path, tempfile_basename| Dir::Tmpname.create(tempfile_basename) do |path| io = File.open(original_file_path) io.seek(0) # rewind the file. This is necessary when testing with a mock StringIO contents = io.read write_file(path, contents) end - end.first + end end def generate_static_html diff --git a/spec/requests/api/markdown_snapshot_spec.rb b/spec/requests/api/markdown_snapshot_spec.rb index 1270efdfd6fd3..f2019172a54fe 100644 --- a/spec/requests/api/markdown_snapshot_spec.rb +++ b/spec/requests/api/markdown_snapshot_spec.rb @@ -5,7 +5,5 @@ # See https://docs.gitlab.com/ee/development/gitlab_flavored_markdown/specification_guide/#markdown-snapshot-testing # for documentation on this spec. RSpec.describe API::Markdown, 'Snapshot' do - # noinspection RubyMismatchedArgumentType (ignore RBS type warning: __dir__ can be nil, but 2nd argument can't be nil) - glfm_specification_dir = File.expand_path('../../../glfm_specification', __dir__) - include_context 'with API::Markdown Snapshot shared context', glfm_specification_dir + include_context 'with API::Markdown Snapshot shared context' end diff --git a/spec/scripts/lib/glfm/parse_examples_spec.rb b/spec/scripts/lib/glfm/parse_examples_spec.rb index 6fe1149a3789d..a1ee6b3f440eb 100644 --- a/spec/scripts/lib/glfm/parse_examples_spec.rb +++ b/spec/scripts/lib/glfm/parse_examples_spec.rb @@ -9,7 +9,7 @@ end let(:spec_txt_contents) do - <<~SPEC_TXT_CONTENTS + <<~MARKDOWN --- title: Spec ... @@ -134,7 +134,7 @@ # Appendix Appendix text. - SPEC_TXT_CONTENTS + MARKDOWN end let(:spec_txt_lines) { spec_txt_contents.split("\n") } @@ -311,7 +311,7 @@ describe "with incorrect header nesting" do let(:spec_txt_contents) do - <<~SPEC_TXT_CONTENTS + <<~MARKDOWN --- title: Spec ... @@ -320,7 +320,7 @@ ### H3 - SPEC_TXT_CONTENTS + MARKDOWN end it "raises if H3 is nested directly in H1" do diff --git a/spec/scripts/lib/glfm/update_example_snapshots_spec.rb b/spec/scripts/lib/glfm/update_example_snapshots_spec.rb index 34af35a66f558..f96936c0a6f2c 100644 --- a/spec/scripts/lib/glfm/update_example_snapshots_spec.rb +++ b/spec/scripts/lib/glfm/update_example_snapshots_spec.rb @@ -35,6 +35,8 @@ let(:glfm_spec_txt_local_io) { StringIO.new(glfm_spec_txt_contents) } let(:glfm_example_status_yml_path) { described_class::GLFM_EXAMPLE_STATUS_YML_PATH } let(:glfm_example_status_yml_io) { StringIO.new(glfm_example_status_yml_contents) } + let(:glfm_example_metadata_yml_path) { described_class::GLFM_EXAMPLE_METADATA_YML_PATH } + let(:glfm_example_metadata_yml_io) { StringIO.new(glfm_example_metadata_yml_contents) } # Example Snapshot (ES) output files let(:es_examples_index_yml_path) { described_class::ES_EXAMPLES_INDEX_YML_PATH } @@ -52,7 +54,7 @@ let(:static_html_tempfile_path) { Tempfile.new.path } let(:glfm_spec_txt_contents) do - <<~GLFM_SPEC_TXT_CONTENTS + <<~MARKDOWN --- title: GitLab Flavored Markdown Spec ... @@ -183,17 +185,75 @@ <p><strong>This example will have its manually modified static HTML, WYSIWYG HTML, and ProseMirror JSON preserved</strong></p> ```````````````````````````````` + # API Request Overrides + + This section contains examples which verify that all of the fixture models which are set up + in `render_static_html.rb` are correctly configured. They exercise various `preview_markdown` + endpoints via `glfm_example_metadata.yml`. + + ## Group Upload Link + + `preview_markdown` exercising `groups` API endpoint and `UploadLinkFilter`: + + ```````````````````````````````` example gitlab + [groups-test-file](/uploads/groups-test-file) + . + <p><a href="groups-test-file">groups-test-file</a></p> + ```````````````````````````````` + + ## Project Repo Link + + `preview_markdown` exercising `projects` API endpoint and `RepositoryLinkFilter`: + + ```````````````````````````````` example gitlab + [projects-test-file](projects-test-file) + . + <p><a href="projects-test-file">projects-test-file</a></p> + ```````````````````````````````` + + ## Project Snippet Ref + + `preview_markdown` exercising `projects` API endpoint and `SnippetReferenceFilter`: + + ```````````````````````````````` example gitlab + This project snippet ID reference IS filtered: $88888 + . + <p>This project snippet ID reference IS filtered: <a href="/glfm_group/glfm_project/-/snippets/88888">$88888</a> + ```````````````````````````````` + + ## Personal Snippet Ref + + `preview_markdown` exercising personal (non-project) `snippets` API endpoint. This is + only used by the comment field on personal snippets. It has no unique custom markdown + extension behavior, and specifically does not render snippet references via + `SnippetReferenceFilter`, even if the ID is valid. + + ```````````````````````````````` example gitlab + This personal snippet ID reference is NOT filtered: $99999 + . + <p>This personal snippet ID reference is NOT filtered: $99999</p> + ```````````````````````````````` + + ## Project Wiki Link + + `preview_markdown` exercising project `wikis` API endpoint and `WikiLinkFilter`: + + ```````````````````````````````` example gitlab + [project-wikis-test-file](project-wikis-test-file) + . + <p><a href="project-wikis-test-file">project-wikis-test-file</a></p> + ```````````````````````````````` + <!-- END TESTS --> # Appendix Appendix text. - GLFM_SPEC_TXT_CONTENTS + MARKDOWN end let(:glfm_example_status_yml_contents) do - # language=YAML - <<~GLFM_EXAMPLE_STATUS_YML_CONTENTS + <<~YAML --- 02_01_00__inlines__strong__001: # The skip_update_example_snapshots key is present, but false, so this example is not skipped @@ -206,12 +266,27 @@ 05_02_00__third_gitlab_specific_section_with_skipped_examples__strong_but_manually_modified_and_skipped__001: # Always skip this example, but preserve the existing manual modifications skip_update_example_snapshots: 'skipping this example because we have manually modified it' - GLFM_EXAMPLE_STATUS_YML_CONTENTS + YAML + end + + let(:glfm_example_metadata_yml_contents) do + <<~YAML + --- + 06_01_00__api_request_overrides__group_upload_link__001: + api_request_override_path: /groups/glfm_group/preview_markdown + 06_02_00__api_request_overrides__project_repo_link__001: + api_request_override_path: /glfm_group/glfm_project/preview_markdown + 06_03_00__api_request_overrides__project_snippet_ref__001: + api_request_override_path: /glfm_group/glfm_project/preview_markdown + 06_04_00__api_request_overrides__personal_snippet_ref__001: + api_request_override_path: /-/snippets/preview_markdown + 06_05_00__api_request_overrides__project_wiki_link__001: + api_request_override_path: /glfm_group/glfm_project/-/wikis/new_page/preview_markdown + YAML end let(:es_html_yml_io_existing_contents) do - # language=YAML - <<~ES_HTML_YML_IO_EXISTING_CONTENTS + <<~YAML --- 00_00_00__obsolete_entry_to_be_deleted__001: canonical: | @@ -234,12 +309,11 @@ <p>This is the manually modified static HTML which will be preserved</p> wysiwyg: |- <p>This is the manually modified WYSIWYG HTML which will be preserved</p> - ES_HTML_YML_IO_EXISTING_CONTENTS + YAML end let(:es_prosemirror_json_yml_io_existing_contents) do - # language=YAML - <<~ES_PROSEMIRROR_JSON_YML_IO_EXISTING_CONTENTS + <<~YAML --- 00_00_00__obsolete_entry_to_be_deleted__001: |- { @@ -272,7 +346,7 @@ { "existing": "This entry is manually modified and preserved because skip_update_example_snapshots will be truthy" } - ES_PROSEMIRROR_JSON_YML_IO_EXISTING_CONTENTS + YAML end before do @@ -283,6 +357,9 @@ # input files allow(File).to receive(:open).with(glfm_spec_txt_path) { glfm_spec_txt_local_io } allow(File).to receive(:open).with(glfm_example_status_yml_path) { glfm_example_status_yml_io } + allow(File).to receive(:open).with(glfm_example_metadata_yml_path) do + glfm_example_metadata_yml_io + end # output files allow(File).to receive(:open).with(es_examples_index_yml_path, 'w') { es_examples_index_yml_io } @@ -298,6 +375,7 @@ # Allow normal opening of Tempfile files created during script execution. tempfile_basenames = [ described_class::MARKDOWN_TEMPFILE_BASENAME[0], + described_class::METADATA_TEMPFILE_BASENAME[0], described_class::STATIC_HTML_TEMPFILE_BASENAME[0], described_class::WYSIWYG_HTML_AND_JSON_TEMPFILE_BASENAME[0] ].join('|') @@ -333,13 +411,12 @@ describe 'when any other skip_update_example_snapshot_* is also truthy' do let(:glfm_example_status_yml_contents) do - # language=YAML - <<~GLFM_EXAMPLE_STATUS_YML_CONTENTS + <<~YAML --- 02_01_00__inlines__strong__001: skip_update_example_snapshots: 'if the skip_update_example_snapshots key is truthy...' skip_update_example_snapshot_html_static: '...then no other skip_update_example_* keys can be truthy' - GLFM_EXAMPLE_STATUS_YML_CONTENTS + YAML end it 'raises an error' do @@ -354,8 +431,7 @@ describe 'writing examples_index.yml' do let(:es_examples_index_yml_contents) { reread_io(es_examples_index_yml_io) } let(:expected_examples_index_yml_contents) do - # language=YAML - <<~ES_EXAMPLES_INDEX_YML_CONTENTS + <<~YAML --- 02_01_00__inlines__strong__001: spec_txt_example_position: 1 @@ -381,7 +457,22 @@ 05_02_00__third_gitlab_specific_section_with_skipped_examples__strong_but_manually_modified_and_skipped__001: spec_txt_example_position: 9 source_specification: gitlab - ES_EXAMPLES_INDEX_YML_CONTENTS + 06_01_00__api_request_overrides__group_upload_link__001: + spec_txt_example_position: 10 + source_specification: gitlab + 06_02_00__api_request_overrides__project_repo_link__001: + spec_txt_example_position: 11 + source_specification: gitlab + 06_03_00__api_request_overrides__project_snippet_ref__001: + spec_txt_example_position: 12 + source_specification: gitlab + 06_04_00__api_request_overrides__personal_snippet_ref__001: + spec_txt_example_position: 13 + source_specification: gitlab + 06_05_00__api_request_overrides__project_wiki_link__001: + spec_txt_example_position: 14 + source_specification: gitlab + YAML end it 'writes the correct content' do @@ -394,8 +485,7 @@ describe 'writing markdown.yml' do let(:es_markdown_yml_contents) { reread_io(es_markdown_yml_io) } let(:expected_markdown_yml_contents) do - # language=YAML - <<~ES_MARKDOWN_YML_CONTENTS + <<~YAML --- 02_01_00__inlines__strong__001: | __bold__ @@ -415,7 +505,17 @@ **this example will be skipped** 05_02_00__third_gitlab_specific_section_with_skipped_examples__strong_but_manually_modified_and_skipped__001: | **This example will have its manually modified static HTML, WYSIWYG HTML, and ProseMirror JSON preserved** - ES_MARKDOWN_YML_CONTENTS + 06_01_00__api_request_overrides__group_upload_link__001: | + [groups-test-file](/uploads/groups-test-file) + 06_02_00__api_request_overrides__project_repo_link__001: | + [projects-test-file](projects-test-file) + 06_03_00__api_request_overrides__project_snippet_ref__001: | + This project snippet ID reference IS filtered: $88888 + 06_04_00__api_request_overrides__personal_snippet_ref__001: | + This personal snippet ID reference is NOT filtered: $99999 + 06_05_00__api_request_overrides__project_wiki_link__001: | + [project-wikis-test-file](project-wikis-test-file) + YAML end it 'writes the correct content' do @@ -425,6 +525,7 @@ end end + # rubocop:disable RSpec/MultipleMemoizedHelpers describe 'writing html.yml and prosemirror_json.yml' do let(:es_html_yml_contents) { reread_io(es_html_yml_io) } let(:es_prosemirror_json_yml_contents) { reread_io(es_prosemirror_json_yml_io) } @@ -432,8 +533,7 @@ # NOTE: This example_status.yml is crafted in conjunction with expected_html_yml_contents # to test the behavior of the `skip_update_*` flags let(:glfm_example_status_yml_contents) do - # language=YAML - <<~GLFM_EXAMPLE_STATUS_YML_CONTENTS + <<~YAML --- 02_01_00__inlines__strong__002: # NOTE: 02_01_00__inlines__strong__002: is omitted from the existing prosemirror_json.yml file, and is also @@ -451,12 +551,11 @@ skip_update_example_snapshots: 'skipping this example because it is very bad' 05_02_00__third_gitlab_specific_section_with_skipped_examples__strong_but_manually_modified_and_skipped__001: skip_update_example_snapshots: 'skipping this example because we have manually modified it' - GLFM_EXAMPLE_STATUS_YML_CONTENTS + YAML end let(:expected_html_yml_contents) do - # language=YAML - <<~ES_HTML_YML_CONTENTS + <<~YAML --- 02_01_00__inlines__strong__001: canonical: | @@ -507,12 +606,46 @@ <p>This is the manually modified static HTML which will be preserved</p> wysiwyg: |- <p>This is the manually modified WYSIWYG HTML which will be preserved</p> - ES_HTML_YML_CONTENTS + 06_01_00__api_request_overrides__group_upload_link__001: + canonical: | + <p><a href="groups-test-file">groups-test-file</a></p> + static: |- + <p data-sourcepos="1:1-1:45" dir="auto"><a href="/groups/glfm_group/-/uploads/groups-test-file" data-canonical-src="/uploads/groups-test-file" data-link="true" class="gfm">groups-test-file</a></p> + wysiwyg: |- + <p><a target="_blank" rel="noopener noreferrer nofollow" href="/uploads/groups-test-file">groups-test-file</a></p> + 06_02_00__api_request_overrides__project_repo_link__001: + canonical: | + <p><a href="projects-test-file">projects-test-file</a></p> + static: |- + <p data-sourcepos="1:1-1:40" dir="auto"><a href="/glfm_group/glfm_project/-/blob/master/projects-test-file">projects-test-file</a></p> + wysiwyg: |- + <p><a target="_blank" rel="noopener noreferrer nofollow" href="projects-test-file">projects-test-file</a></p> + 06_03_00__api_request_overrides__project_snippet_ref__001: + canonical: | + <p>This project snippet ID reference IS filtered: <a href="/glfm_group/glfm_project/-/snippets/88888">$88888</a> + static: |- + <p data-sourcepos="1:1-1:53" dir="auto">This project snippet ID reference IS filtered: <a href="/glfm_group/glfm_project/-/snippets/88888" data-reference-type="snippet" data-original="$88888" data-link="false" data-link-reference="false" data-project="77777" data-snippet="88888" data-container="body" data-placement="top" title="glfm_project_snippet" class="gfm gfm-snippet has-tooltip">$88888</a></p> + wysiwyg: |- + <p>This project snippet ID reference IS filtered: $88888</p> + 06_04_00__api_request_overrides__personal_snippet_ref__001: + canonical: | + <p>This personal snippet ID reference is NOT filtered: $99999</p> + static: |- + <p data-sourcepos="1:1-1:58" dir="auto">This personal snippet ID reference is NOT filtered: $99999</p> + wysiwyg: |- + <p>This personal snippet ID reference is NOT filtered: $99999</p> + 06_05_00__api_request_overrides__project_wiki_link__001: + canonical: | + <p><a href="project-wikis-test-file">project-wikis-test-file</a></p> + static: |- + <p data-sourcepos="1:1-1:50" dir="auto"><a href="/glfm_group/glfm_project/-/wikis/project-wikis-test-file" data-canonical-src="project-wikis-test-file">project-wikis-test-file</a></p> + wysiwyg: |- + <p><a target="_blank" rel="noopener noreferrer nofollow" href="project-wikis-test-file">project-wikis-test-file</a></p> + YAML end let(:expected_prosemirror_json_contents) do - # language=YAML - <<~ES_PROSEMIRROR_JSON_YML_CONTENTS + <<~YAML --- 02_01_00__inlines__strong__001: |- { @@ -601,7 +734,121 @@ { "existing": "This entry is manually modified and preserved because skip_update_example_snapshots will be truthy" } - ES_PROSEMIRROR_JSON_YML_CONTENTS + 06_01_00__api_request_overrides__group_upload_link__001: |- + { + "type": "doc", + "content": [ + { + "type": "paragraph", + "content": [ + { + "type": "text", + "marks": [ + { + "type": "link", + "attrs": { + "href": "/uploads/groups-test-file", + "target": "_blank", + "class": null, + "title": null, + "canonicalSrc": "/uploads/groups-test-file", + "isReference": false + } + } + ], + "text": "groups-test-file" + } + ] + } + ] + } + 06_02_00__api_request_overrides__project_repo_link__001: |- + { + "type": "doc", + "content": [ + { + "type": "paragraph", + "content": [ + { + "type": "text", + "marks": [ + { + "type": "link", + "attrs": { + "href": "projects-test-file", + "target": "_blank", + "class": null, + "title": null, + "canonicalSrc": "projects-test-file", + "isReference": false + } + } + ], + "text": "projects-test-file" + } + ] + } + ] + } + 06_03_00__api_request_overrides__project_snippet_ref__001: |- + { + "type": "doc", + "content": [ + { + "type": "paragraph", + "content": [ + { + "type": "text", + "text": "This project snippet ID reference IS filtered: $88888" + } + ] + } + ] + } + 06_04_00__api_request_overrides__personal_snippet_ref__001: |- + { + "type": "doc", + "content": [ + { + "type": "paragraph", + "content": [ + { + "type": "text", + "text": "This personal snippet ID reference is NOT filtered: $99999" + } + ] + } + ] + } + 06_05_00__api_request_overrides__project_wiki_link__001: |- + { + "type": "doc", + "content": [ + { + "type": "paragraph", + "content": [ + { + "type": "text", + "marks": [ + { + "type": "link", + "attrs": { + "href": "project-wikis-test-file", + "target": "_blank", + "class": null, + "title": null, + "canonicalSrc": "project-wikis-test-file", + "isReference": false + } + } + ], + "text": "project-wikis-test-file" + } + ] + } + ] + } + YAML end before do @@ -627,6 +874,7 @@ expect(es_prosemirror_json_yml_contents).to eq(expected_prosemirror_json_contents) end end + # rubocop:enable RSpec/MultipleMemoizedHelpers def reread_io(io) # Reset the io StringIO to the beginning position of the buffer diff --git a/spec/scripts/lib/glfm/update_specification_spec.rb b/spec/scripts/lib/glfm/update_specification_spec.rb index e8d34b13efa54..9fb671e0016a1 100644 --- a/spec/scripts/lib/glfm/update_specification_spec.rb +++ b/spec/scripts/lib/glfm/update_specification_spec.rb @@ -18,7 +18,7 @@ let(:glfm_spec_txt_io) { StringIO.new } let(:ghfm_spec_txt_contents) do - <<~GHFM_SPEC_TXT_CONTENTS + <<~MARKDOWN --- title: GitHub Flavored Markdown Spec version: 0.29 @@ -49,25 +49,26 @@ # Appendix Appendix text. - GHFM_SPEC_TXT_CONTENTS + MARKDOWN end let(:glfm_intro_txt_contents) do - <<~GLFM_INTRO_TXT_CONTENTS + # language=Markdown + <<~MARKDOWN # Introduction ## What is GitLab Flavored Markdown? Intro text about GitLab Flavored Markdown. - GLFM_INTRO_TXT_CONTENTS + MARKDOWN end let(:glfm_examples_txt_contents) do - <<~GLFM_EXAMPLES_TXT_CONTENTS + <<~MARKDOWN # GitLab-Specific Section with Examples Some examples. - GLFM_EXAMPLES_TXT_CONTENTS + MARKDOWN end before do @@ -118,12 +119,12 @@ context 'with error handling' do context 'with a version mismatch' do let(:ghfm_spec_txt_contents) do - <<~GHFM_SPEC_TXT_CONTENTS + <<~MARKDOWN --- title: GitHub Flavored Markdown Spec version: 0.30 ... - GHFM_SPEC_TXT_CONTENTS + MARKDOWN end it 'raises an error' do @@ -173,7 +174,7 @@ end it 'inserts the GitLab examples sections before the appendix section' do - expected = <<~GHFM_SPEC_TXT_CONTENTS + expected = <<~MARKDOWN End of last GitHub examples section. # GitLab-Specific Section with Examples @@ -183,7 +184,7 @@ <!-- END TESTS --> # Appendix - GHFM_SPEC_TXT_CONTENTS + MARKDOWN expect(glfm_contents).to match(/#{Regexp.escape(expected)}/m) end end diff --git a/spec/support/shared_contexts/markdown_snapshot_shared_examples.rb b/spec/support/shared_contexts/glfm/api_markdown_snapshot_shared_context.rb similarity index 66% rename from spec/support/shared_contexts/markdown_snapshot_shared_examples.rb rename to spec/support/shared_contexts/glfm/api_markdown_snapshot_shared_context.rb index 8401ab6541811..3623fa0850d8f 100644 --- a/spec/support/shared_contexts/markdown_snapshot_shared_examples.rb +++ b/spec/support/shared_contexts/glfm/api_markdown_snapshot_shared_context.rb @@ -1,37 +1,28 @@ # frozen_string_literal: true -require 'spec_helper' +require_relative '../../../../scripts/lib/glfm/constants' # See https://docs.gitlab.com/ee/development/gitlab_flavored_markdown/specification_guide/#markdown-snapshot-testing # for documentation on this spec. -RSpec.shared_context 'with API::Markdown Snapshot shared context' do |glfm_specification_dir| - include ApiHelpers - - let_it_be(:user) { create(:user) } - let_it_be(:api_url) { api('/markdown', user) } - - before do - # Set 'GITLAB_TEST_FOOTNOTE_ID' in order to override random number generation in - # Banzai::Filter::FootnoteFilter#random_number, and thus avoid the need to - # perform normalization on the value. See: - # https://docs.gitlab.com/ee/development/gitlab_flavored_markdown/specification_guide/#normalization - stub_env('GITLAB_TEST_FOOTNOTE_ID', 42) - end +RSpec.shared_context 'with API::Markdown Snapshot shared context' do |ee_only: false| + include_context 'with GLFM example snapshot fixtures' - markdown_examples, html_examples = %w[markdown.yml html.yml].map do |file_name| - yaml = File.read("#{glfm_specification_dir}/example_snapshots/#{file_name}") - YAML.safe_load(yaml, symbolize_names: true, aliases: true) - end + include ApiHelpers - normalizations_yaml = File.read( - "#{glfm_specification_dir}/input/gitlab_flavored_markdown/glfm_example_normalizations.yml") - normalizations_by_example_name = YAML.safe_load(normalizations_yaml, symbolize_names: true, aliases: true) + markdown_examples, html_examples, normalizations_by_example_name, metadata_by_example_name = [ + Glfm::Constants::ES_MARKDOWN_YML_PATH, + Glfm::Constants::ES_HTML_YML_PATH, + Glfm::Constants::GLFM_EXAMPLE_NORMALIZATIONS_YML_PATH, + Glfm::Constants::GLFM_EXAMPLE_METADATA_YML_PATH + ].map { |path| YAML.safe_load(File.open(path), symbolize_names: true, aliases: true) } if (focused_markdown_examples_string = ENV['FOCUSED_MARKDOWN_EXAMPLES']) focused_markdown_examples = focused_markdown_examples_string.split(',').map(&:strip).map(&:to_sym) markdown_examples.select! { |example_name| focused_markdown_examples.include?(example_name) } end + markdown_examples.select! { |example_name| !!metadata_by_example_name&.dig(example_name, :ee) == ee_only } + markdown_examples.each do |name, markdown| context "for #{name}" do let(:html) { html_examples.fetch(name).fetch(:static) } @@ -40,6 +31,7 @@ it "verifies conversion of GLFM to HTML", :unlimited_max_formatted_output_length do # noinspection RubyResolve normalized_html = normalize_html(html, normalizations) + api_url = metadata_by_example_name&.dig(name, :api_request_override_path) || (api "/markdown") post api_url, params: { text: markdown, gfm: true } expect(response).to be_successful diff --git a/spec/support/shared_contexts/glfm/example_snapshot_fixtures.rb b/spec/support/shared_contexts/glfm/example_snapshot_fixtures.rb new file mode 100644 index 0000000000000..22b401bc8413e --- /dev/null +++ b/spec/support/shared_contexts/glfm/example_snapshot_fixtures.rb @@ -0,0 +1,27 @@ +# frozen_string_literal: true + +RSpec.shared_context 'with GLFM example snapshot fixtures' do + let_it_be(:user) { create(:user) } + let_it_be(:group) { create(:group, name: 'glfm_group').tap { |group| group.add_owner(user) } } + + let_it_be(:project) do + # NOTE: We hardcode the IDs on all fixtures to prevent variability in the + # rendered HTML/Prosemirror JSON, and to minimize the need for normalization: + # https://docs.gitlab.com/ee/development/gitlab_flavored_markdown/specification_guide/#normalization + create(:project, :repository, creator: user, group: group, name: 'glfm_project', id: 77777) + end + + let_it_be(:project_snippet) { create(:project_snippet, title: 'glfm_project_snippet', id: 88888, project: project) } + let_it_be(:personal_snippet) { create(:snippet, id: 99999) } + + before do + # Set 'GITLAB_TEST_FOOTNOTE_ID' in order to override random number generation in + # Banzai::Filter::FootnoteFilter#random_number, and thus avoid the need to + # perform normalization on the value. See: + # https://docs.gitlab.com/ee/development/gitlab_flavored_markdown/specification_guide/#normalization + stub_env('GITLAB_TEST_FOOTNOTE_ID', 42) + + stub_licensed_features(group_wikis: true) + sign_in(user) + end +end -- GitLab