smart_scheduling_links_test_kit 0.1.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/LICENSE +201 -0
- data/lib/smart_scheduling_links_test_kit/manifest_group.rb +223 -0
- data/lib/smart_scheduling_links_test_kit/resource_group.rb +154 -0
- data/lib/smart_scheduling_links_test_kit/version.rb +3 -0
- data/lib/smart_scheduling_links_test_kit.rb +72 -0
- metadata +120 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: 5fb516e68d836499cdcd723921876e2a6cf5aedf62dce8ee75ac6cb3eef1691a
|
4
|
+
data.tar.gz: 5729dbc340395dee715d0026434604059827e038e01de7821e9736e4f3af164f
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: 7f1cff98a95d5fe83aabc49705f21224eb1b41a8d905d0d5177c744c8b2d9fa2ab8148138b4382693d41fb01bcf438be3e4c3914a22144fbf493bbe082771500
|
7
|
+
data.tar.gz: 1119b5427ccb04b5d4b36537fec5f081821900fc83d8c67a1b85b5eb75891e49bc85d11a1352562029dd3a3796db5975eb7e72f5cbfd44c8f9f779a20303fb45
|
data/LICENSE
ADDED
@@ -0,0 +1,201 @@
|
|
1
|
+
Apache License
|
2
|
+
Version 2.0, January 2004
|
3
|
+
http://www.apache.org/licenses/
|
4
|
+
|
5
|
+
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
6
|
+
|
7
|
+
1. Definitions.
|
8
|
+
|
9
|
+
"License" shall mean the terms and conditions for use, reproduction,
|
10
|
+
and distribution as defined by Sections 1 through 9 of this document.
|
11
|
+
|
12
|
+
"Licensor" shall mean the copyright owner or entity authorized by
|
13
|
+
the copyright owner that is granting the License.
|
14
|
+
|
15
|
+
"Legal Entity" shall mean the union of the acting entity and all
|
16
|
+
other entities that control, are controlled by, or are under common
|
17
|
+
control with that entity. For the purposes of this definition,
|
18
|
+
"control" means (i) the power, direct or indirect, to cause the
|
19
|
+
direction or management of such entity, whether by contract or
|
20
|
+
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
21
|
+
outstanding shares, or (iii) beneficial ownership of such entity.
|
22
|
+
|
23
|
+
"You" (or "Your") shall mean an individual or Legal Entity
|
24
|
+
exercising permissions granted by this License.
|
25
|
+
|
26
|
+
"Source" form shall mean the preferred form for making modifications,
|
27
|
+
including but not limited to software source code, documentation
|
28
|
+
source, and configuration files.
|
29
|
+
|
30
|
+
"Object" form shall mean any form resulting from mechanical
|
31
|
+
transformation or translation of a Source form, including but
|
32
|
+
not limited to compiled object code, generated documentation,
|
33
|
+
and conversions to other media types.
|
34
|
+
|
35
|
+
"Work" shall mean the work of authorship, whether in Source or
|
36
|
+
Object form, made available under the License, as indicated by a
|
37
|
+
copyright notice that is included in or attached to the work
|
38
|
+
(an example is provided in the Appendix below).
|
39
|
+
|
40
|
+
"Derivative Works" shall mean any work, whether in Source or Object
|
41
|
+
form, that is based on (or derived from) the Work and for which the
|
42
|
+
editorial revisions, annotations, elaborations, or other modifications
|
43
|
+
represent, as a whole, an original work of authorship. For the purposes
|
44
|
+
of this License, Derivative Works shall not include works that remain
|
45
|
+
separable from, or merely link (or bind by name) to the interfaces of,
|
46
|
+
the Work and Derivative Works thereof.
|
47
|
+
|
48
|
+
"Contribution" shall mean any work of authorship, including
|
49
|
+
the original version of the Work and any modifications or additions
|
50
|
+
to that Work or Derivative Works thereof, that is intentionally
|
51
|
+
submitted to Licensor for inclusion in the Work by the copyright owner
|
52
|
+
or by an individual or Legal Entity authorized to submit on behalf of
|
53
|
+
the copyright owner. For the purposes of this definition, "submitted"
|
54
|
+
means any form of electronic, verbal, or written communication sent
|
55
|
+
to the Licensor or its representatives, including but not limited to
|
56
|
+
communication on electronic mailing lists, source code control systems,
|
57
|
+
and issue tracking systems that are managed by, or on behalf of, the
|
58
|
+
Licensor for the purpose of discussing and improving the Work, but
|
59
|
+
excluding communication that is conspicuously marked or otherwise
|
60
|
+
designated in writing by the copyright owner as "Not a Contribution."
|
61
|
+
|
62
|
+
"Contributor" shall mean Licensor and any individual or Legal Entity
|
63
|
+
on behalf of whom a Contribution has been received by Licensor and
|
64
|
+
subsequently incorporated within the Work.
|
65
|
+
|
66
|
+
2. Grant of Copyright License. Subject to the terms and conditions of
|
67
|
+
this License, each Contributor hereby grants to You a perpetual,
|
68
|
+
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
69
|
+
copyright license to reproduce, prepare Derivative Works of,
|
70
|
+
publicly display, publicly perform, sublicense, and distribute the
|
71
|
+
Work and such Derivative Works in Source or Object form.
|
72
|
+
|
73
|
+
3. Grant of Patent License. Subject to the terms and conditions of
|
74
|
+
this License, each Contributor hereby grants to You a perpetual,
|
75
|
+
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
76
|
+
(except as stated in this section) patent license to make, have made,
|
77
|
+
use, offer to sell, sell, import, and otherwise transfer the Work,
|
78
|
+
where such license applies only to those patent claims licensable
|
79
|
+
by such Contributor that are necessarily infringed by their
|
80
|
+
Contribution(s) alone or by combination of their Contribution(s)
|
81
|
+
with the Work to which such Contribution(s) was submitted. If You
|
82
|
+
institute patent litigation against any entity (including a
|
83
|
+
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
84
|
+
or a Contribution incorporated within the Work constitutes direct
|
85
|
+
or contributory patent infringement, then any patent licenses
|
86
|
+
granted to You under this License for that Work shall terminate
|
87
|
+
as of the date such litigation is filed.
|
88
|
+
|
89
|
+
4. Redistribution. You may reproduce and distribute copies of the
|
90
|
+
Work or Derivative Works thereof in any medium, with or without
|
91
|
+
modifications, and in Source or Object form, provided that You
|
92
|
+
meet the following conditions:
|
93
|
+
|
94
|
+
(a) You must give any other recipients of the Work or
|
95
|
+
Derivative Works a copy of this License; and
|
96
|
+
|
97
|
+
(b) You must cause any modified files to carry prominent notices
|
98
|
+
stating that You changed the files; and
|
99
|
+
|
100
|
+
(c) You must retain, in the Source form of any Derivative Works
|
101
|
+
that You distribute, all copyright, patent, trademark, and
|
102
|
+
attribution notices from the Source form of the Work,
|
103
|
+
excluding those notices that do not pertain to any part of
|
104
|
+
the Derivative Works; and
|
105
|
+
|
106
|
+
(d) If the Work includes a "NOTICE" text file as part of its
|
107
|
+
distribution, then any Derivative Works that You distribute must
|
108
|
+
include a readable copy of the attribution notices contained
|
109
|
+
within such NOTICE file, excluding those notices that do not
|
110
|
+
pertain to any part of the Derivative Works, in at least one
|
111
|
+
of the following places: within a NOTICE text file distributed
|
112
|
+
as part of the Derivative Works; within the Source form or
|
113
|
+
documentation, if provided along with the Derivative Works; or,
|
114
|
+
within a display generated by the Derivative Works, if and
|
115
|
+
wherever such third-party notices normally appear. The contents
|
116
|
+
of the NOTICE file are for informational purposes only and
|
117
|
+
do not modify the License. You may add Your own attribution
|
118
|
+
notices within Derivative Works that You distribute, alongside
|
119
|
+
or as an addendum to the NOTICE text from the Work, provided
|
120
|
+
that such additional attribution notices cannot be construed
|
121
|
+
as modifying the License.
|
122
|
+
|
123
|
+
You may add Your own copyright statement to Your modifications and
|
124
|
+
may provide additional or different license terms and conditions
|
125
|
+
for use, reproduction, or distribution of Your modifications, or
|
126
|
+
for any such Derivative Works as a whole, provided Your use,
|
127
|
+
reproduction, and distribution of the Work otherwise complies with
|
128
|
+
the conditions stated in this License.
|
129
|
+
|
130
|
+
5. Submission of Contributions. Unless You explicitly state otherwise,
|
131
|
+
any Contribution intentionally submitted for inclusion in the Work
|
132
|
+
by You to the Licensor shall be under the terms and conditions of
|
133
|
+
this License, without any additional terms or conditions.
|
134
|
+
Notwithstanding the above, nothing herein shall supersede or modify
|
135
|
+
the terms of any separate license agreement you may have executed
|
136
|
+
with Licensor regarding such Contributions.
|
137
|
+
|
138
|
+
6. Trademarks. This License does not grant permission to use the trade
|
139
|
+
names, trademarks, service marks, or product names of the Licensor,
|
140
|
+
except as required for reasonable and customary use in describing the
|
141
|
+
origin of the Work and reproducing the content of the NOTICE file.
|
142
|
+
|
143
|
+
7. Disclaimer of Warranty. Unless required by applicable law or
|
144
|
+
agreed to in writing, Licensor provides the Work (and each
|
145
|
+
Contributor provides its Contributions) on an "AS IS" BASIS,
|
146
|
+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
147
|
+
implied, including, without limitation, any warranties or conditions
|
148
|
+
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
149
|
+
PARTICULAR PURPOSE. You are solely responsible for determining the
|
150
|
+
appropriateness of using or redistributing the Work and assume any
|
151
|
+
risks associated with Your exercise of permissions under this License.
|
152
|
+
|
153
|
+
8. Limitation of Liability. In no event and under no legal theory,
|
154
|
+
whether in tort (including negligence), contract, or otherwise,
|
155
|
+
unless required by applicable law (such as deliberate and grossly
|
156
|
+
negligent acts) or agreed to in writing, shall any Contributor be
|
157
|
+
liable to You for damages, including any direct, indirect, special,
|
158
|
+
incidental, or consequential damages of any character arising as a
|
159
|
+
result of this License or out of the use or inability to use the
|
160
|
+
Work (including but not limited to damages for loss of goodwill,
|
161
|
+
work stoppage, computer failure or malfunction, or any and all
|
162
|
+
other commercial damages or losses), even if such Contributor
|
163
|
+
has been advised of the possibility of such damages.
|
164
|
+
|
165
|
+
9. Accepting Warranty or Additional Liability. While redistributing
|
166
|
+
the Work or Derivative Works thereof, You may choose to offer,
|
167
|
+
and charge a fee for, acceptance of support, warranty, indemnity,
|
168
|
+
or other liability obligations and/or rights consistent with this
|
169
|
+
License. However, in accepting such obligations, You may act only
|
170
|
+
on Your own behalf and on Your sole responsibility, not on behalf
|
171
|
+
of any other Contributor, and only if You agree to indemnify,
|
172
|
+
defend, and hold each Contributor harmless for any liability
|
173
|
+
incurred by, or claims asserted against, such Contributor by reason
|
174
|
+
of your accepting any such warranty or additional liability.
|
175
|
+
|
176
|
+
END OF TERMS AND CONDITIONS
|
177
|
+
|
178
|
+
APPENDIX: How to apply the Apache License to your work.
|
179
|
+
|
180
|
+
To apply the Apache License to your work, attach the following
|
181
|
+
boilerplate notice, with the fields enclosed by brackets "{}"
|
182
|
+
replaced with your own identifying information. (Don't include
|
183
|
+
the brackets!) The text should be enclosed in the appropriate
|
184
|
+
comment syntax for the file format. We also recommend that a
|
185
|
+
file or class name and description of purpose be included on the
|
186
|
+
same "printed page" as the copyright notice for easier
|
187
|
+
identification within third-party archives.
|
188
|
+
|
189
|
+
Copyright {yyyy} {name of copyright owner}
|
190
|
+
|
191
|
+
Licensed under the Apache License, Version 2.0 (the "License");
|
192
|
+
you may not use this file except in compliance with the License.
|
193
|
+
You may obtain a copy of the License at
|
194
|
+
|
195
|
+
http://www.apache.org/licenses/LICENSE-2.0
|
196
|
+
|
197
|
+
Unless required by applicable law or agreed to in writing, software
|
198
|
+
distributed under the License is distributed on an "AS IS" BASIS,
|
199
|
+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
200
|
+
See the License for the specific language governing permissions and
|
201
|
+
limitations under the License.
|
@@ -0,0 +1,223 @@
|
|
1
|
+
module SMARTSchedulingLinks
|
2
|
+
class ManifestGroup < Inferno::TestGroup
|
3
|
+
id :smart_scheduling_links_manifest
|
4
|
+
title 'Bulk Publication Manifest'
|
5
|
+
description %(
|
6
|
+
This group of tests retrieves a bulk publication manifest and verifies
|
7
|
+
that it is properly structured.
|
8
|
+
)
|
9
|
+
|
10
|
+
input :url,
|
11
|
+
title: 'Bulk Publication Manifest Url'
|
12
|
+
|
13
|
+
test do
|
14
|
+
id :manifest_url_form
|
15
|
+
title 'Bulk Publication Manifest is a valid URL ending in $bulk-publish'
|
16
|
+
description %(
|
17
|
+
The manifest is always hosted at a URL that ends with `$bulk-publish`
|
18
|
+
|
19
|
+
https://github.com/smart-on-fhir/smart-scheduling-links/blob/master/specification.md#quick-start-guide
|
20
|
+
)
|
21
|
+
|
22
|
+
run do
|
23
|
+
assert url.ends_with?('$bulk-publish'), "`#{url}` does not end with `$bulk-publish`"
|
24
|
+
|
25
|
+
assert_valid_http_uri(url)
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
test do
|
30
|
+
id :manifest_download
|
31
|
+
title 'Bulk Publication Manifest can be downloaded'
|
32
|
+
description %(
|
33
|
+
Verify that the manifest can be downloaded and is valid JSON.
|
34
|
+
|
35
|
+
Note that the spec requires that servers respond the same way if a
|
36
|
+
client sends no `Accept` header or an `Accept` header of
|
37
|
+
`application/json`, but this test only tests with the `Accept` header.
|
38
|
+
The HTTP client used by Inferno automatically adds an `Accept` header of
|
39
|
+
`*/*` if no `Accept` header is present. For this reason, Inferno is
|
40
|
+
unable to test that the server correctly responds to a manifest request
|
41
|
+
when no `Accept` header is present.
|
42
|
+
)
|
43
|
+
|
44
|
+
makes_request :manifest
|
45
|
+
output :manifest_json
|
46
|
+
|
47
|
+
run do
|
48
|
+
assert_valid_http_uri(url)
|
49
|
+
|
50
|
+
get url, name: :manifest, headers: { 'Accept' => 'application/json' }
|
51
|
+
assert_response_status(200)
|
52
|
+
assert_valid_json(request.response_body)
|
53
|
+
|
54
|
+
output manifest_json: request.response_body
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
test do
|
59
|
+
id :manifest_cache_control_header
|
60
|
+
title 'Slot Publisher includes a Cache-Control: max-age Header'
|
61
|
+
description %(
|
62
|
+
Slot Publishers SHOULD include a `Cache-Control: max-age=<seconds>`
|
63
|
+
header as a hint to clients about how long (in seconds) to wait before
|
64
|
+
polling next. For example, `Cache-Control: max-age=300` indicates a
|
65
|
+
preferred polling interval of five minutes.
|
66
|
+
)
|
67
|
+
optional
|
68
|
+
|
69
|
+
uses_request :manifest
|
70
|
+
|
71
|
+
run do
|
72
|
+
assert_response_status(200)
|
73
|
+
|
74
|
+
cache_control_header = request.response_header('cache-control')&.value
|
75
|
+
assert cache_control_header&.match?(/max-age=\d+/),
|
76
|
+
"No `Cache-Control: max-age=<seconds>` header received."
|
77
|
+
end
|
78
|
+
end
|
79
|
+
|
80
|
+
test do
|
81
|
+
id :manifest_structure
|
82
|
+
title 'Bulk Publication Manifest has correct structure'
|
83
|
+
description %(
|
84
|
+
Verify that the manifest has the correct JSON structure
|
85
|
+
)
|
86
|
+
|
87
|
+
input :manifest_json, type: 'textarea'
|
88
|
+
output :locations_json, :schedules_json, :slots_json
|
89
|
+
|
90
|
+
INSTANT_REGEX = /([0-9]([0-9]([0-9][1-9]|[1-9]0)|[1-9]00)|[1-9]000)-(0[1-9]|1[0-2])-(0[1-9]|[1-2][0-9]|3[0-1])T([01][0-9]|2[0-3]):[0-5][0-9]:([0-5][0-9]|60)(\.[0-9]+)?(Z|(\+|-)((0[0-9]|1[0-3]):[0-5][0-9]|14:00))/
|
91
|
+
|
92
|
+
run do
|
93
|
+
skip_if manifest_json.blank?, 'No manifest received'
|
94
|
+
assert_valid_json(manifest_json)
|
95
|
+
|
96
|
+
manifest = JSON.parse(manifest_json)
|
97
|
+
assert manifest.is_a?(Hash), "Expected manifest to be a JSON object, but found `#{manifest.class}`"
|
98
|
+
|
99
|
+
manifest_fields = {
|
100
|
+
'transactionTime' => String,
|
101
|
+
'request' => String,
|
102
|
+
'output' => Array
|
103
|
+
}
|
104
|
+
|
105
|
+
manifest_fields.each do |field, type|
|
106
|
+
assert manifest[field].is_a?(type),
|
107
|
+
"`#{field}` field should be `#{type}`, but found `#{manifest[field].class}`"
|
108
|
+
end
|
109
|
+
|
110
|
+
outputs = manifest['output']
|
111
|
+
locations =
|
112
|
+
outputs
|
113
|
+
.select { |output| output['type'] == 'Location' }
|
114
|
+
.map { |output| output['url'] }
|
115
|
+
schedules =
|
116
|
+
outputs
|
117
|
+
.select { |output| output['type'] == 'Schedule' }
|
118
|
+
.map { |output| output['url'] }
|
119
|
+
slots =
|
120
|
+
outputs
|
121
|
+
.select { |output| output['type'] == 'Slot' }
|
122
|
+
.map { |output| output['url'] }
|
123
|
+
|
124
|
+
output locations_json: locations.to_json,
|
125
|
+
schedules_json: schedules.to_json,
|
126
|
+
slots_json: slots.to_json
|
127
|
+
|
128
|
+
transaction_time = manifest['transactionTime']
|
129
|
+
assert transaction_time.match?(INSTANT_REGEX),
|
130
|
+
"`transactionTime` is not in `YYYY-MM-DDThh:mm:ss.sss+zz:zz` format: `#{transaction_time}`"
|
131
|
+
|
132
|
+
request_url = manifest['request']
|
133
|
+
assert_valid_http_uri(request_url)
|
134
|
+
|
135
|
+
manifest_outputs =
|
136
|
+
manifest['output']
|
137
|
+
.reject { |output| ['Location', 'Schedule', 'Slot'].exclude? output['type'] }
|
138
|
+
|
139
|
+
output_fields = {
|
140
|
+
'type' => String,
|
141
|
+
'url' => String
|
142
|
+
}
|
143
|
+
|
144
|
+
manifest_outputs.each do |output|
|
145
|
+
output_fields. each do |field, type|
|
146
|
+
assert output[field].is_a?(type),
|
147
|
+
"`output.#{field}` field should be `#{type}`, but found `#{output[field].class}`"
|
148
|
+
end
|
149
|
+
|
150
|
+
assert_valid_http_uri(output['url'])
|
151
|
+
|
152
|
+
extension = output['extension']
|
153
|
+
if extension.present?
|
154
|
+
assert extension.is_a?(Hash),
|
155
|
+
"`output.extension` field should be a JSON Object, but found `#{extension.class}`"
|
156
|
+
|
157
|
+
states = extension['state']
|
158
|
+
if states.present?
|
159
|
+
assert states.is_a?(Array),
|
160
|
+
"`output.extension.state` field should be an Array, but found `#{states.class}`"
|
161
|
+
|
162
|
+
bad_states = states.reject { |state| state.is_a? String }
|
163
|
+
bad_states_string = bad_states.map { |bad_state| "`#{bad_state.inspect}`" }.join(', ')
|
164
|
+
assert bad_states.empty?,
|
165
|
+
"The following `output.extension.state` entries are not Strings: #{bad_states_string}"
|
166
|
+
end
|
167
|
+
end
|
168
|
+
end
|
169
|
+
end
|
170
|
+
end
|
171
|
+
|
172
|
+
test do
|
173
|
+
id :manifest_state_extensions
|
174
|
+
title 'Slot Publisher annotates each output with a list of states or jurisdictions'
|
175
|
+
description %(
|
176
|
+
Slot Publishers SHOULD annotate each output with a list of states or
|
177
|
+
jurisdictions as a hint to clients, allowing clients to focus on
|
178
|
+
fetching data for the specific states or geographical regions where they
|
179
|
+
operate; this is helpful for clients with limited regions of interest.
|
180
|
+
)
|
181
|
+
optional
|
182
|
+
|
183
|
+
input :manifest_json, type: 'textarea'
|
184
|
+
|
185
|
+
run do
|
186
|
+
skip_if manifest_json.blank?, 'No manifest received'
|
187
|
+
assert_valid_json(manifest_json)
|
188
|
+
|
189
|
+
manifest = JSON.parse(manifest_json)
|
190
|
+
assert manifest.is_a?(Hash), "Expected manifest to be a JSON object, but found `#{manifest.class}`"
|
191
|
+
|
192
|
+
manifest_outputs = manifest['output']
|
193
|
+
assert manifest_outputs.is_a?(Array),
|
194
|
+
"`output` field should be an `Array`, but found `#{manifest_outputs.class}`"
|
195
|
+
|
196
|
+
assert manifest_outputs.all? { |output| output.is_a?(Hash) },
|
197
|
+
"Not all manifest outputs are JSON objects."
|
198
|
+
|
199
|
+
resource_types = ['Location', 'Schedule', 'Slot']
|
200
|
+
|
201
|
+
manifest_outputs.select! { |output| resource_types.include? output['type'] }
|
202
|
+
|
203
|
+
output_counts = Hash.new { |hash, key| hash[key] = 0 }
|
204
|
+
state_extension_counts = Hash.new { |hash, key| hash[key] = 0 }
|
205
|
+
|
206
|
+
manifest_outputs.each do |output|
|
207
|
+
output_counts[output['type']] += 1
|
208
|
+
state_extension_counts[output['type']] += 1 if output.dig('extension', 'state').present?
|
209
|
+
end
|
210
|
+
|
211
|
+
summary =
|
212
|
+
resource_types
|
213
|
+
.map { |type| "* #{state_extension_counts[type]}/#{output_counts[type]} #{type} outputs included `state` extension" }
|
214
|
+
.join("\n")
|
215
|
+
|
216
|
+
assert resource_types.all? { |type| output_counts[type] == state_extension_counts[type] },
|
217
|
+
"Not all outputs included the `state` extension: \n#{summary}"
|
218
|
+
|
219
|
+
pass "All outputs included the `state` extension: \n#{summary}"
|
220
|
+
end
|
221
|
+
end
|
222
|
+
end
|
223
|
+
end
|
@@ -0,0 +1,154 @@
|
|
1
|
+
module SMARTSchedulingLinks
|
2
|
+
module ResourceChecker
|
3
|
+
def max_lines
|
4
|
+
@max_lines ||=
|
5
|
+
max_lines_per_file.to_i == 0 ? 100 : max_lines_per_file.to_i
|
6
|
+
end
|
7
|
+
|
8
|
+
def validate_response(url, profile_url)
|
9
|
+
previous_chunk = String.new
|
10
|
+
|
11
|
+
@valid_resource_count ||= 0
|
12
|
+
|
13
|
+
line_count = 0
|
14
|
+
|
15
|
+
process_body = lambda do |new_chunk, _|
|
16
|
+
current_chunk = previous_chunk + new_chunk
|
17
|
+
lines = current_chunk.lines
|
18
|
+
previous_chunk = lines.pop || String.new
|
19
|
+
|
20
|
+
lines.each do |line|
|
21
|
+
break if (max_lines.present? && line_count >= max_lines) ||
|
22
|
+
messages.any? { |message| message[:type] == 'error' }
|
23
|
+
|
24
|
+
assert_valid_json(line)
|
25
|
+
|
26
|
+
resource = FHIR.from_contents(line)
|
27
|
+
@valid_resource_count += 1 if resource_is_valid?(resource:, profile_url:)
|
28
|
+
|
29
|
+
line_count += 1
|
30
|
+
end
|
31
|
+
end
|
32
|
+
|
33
|
+
stream(process_body, url, headers: { 'Accept' => 'application/fhir+ndjson' })
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
class ResourceGroup < Inferno::TestGroup
|
38
|
+
id :smart_scheduling_links_resources
|
39
|
+
title 'Resource Retrieval'
|
40
|
+
description %(
|
41
|
+
Retrieve and validate the Location, Slot, and Schedule resources listed in
|
42
|
+
a bulk publication manifest.
|
43
|
+
)
|
44
|
+
|
45
|
+
input :max_lines_per_file,
|
46
|
+
title: 'Maximum number of resources to validate per file',
|
47
|
+
default: '100'
|
48
|
+
|
49
|
+
test do
|
50
|
+
include ResourceChecker
|
51
|
+
id :location_resources
|
52
|
+
title 'Location Resources'
|
53
|
+
|
54
|
+
input :locations_json,
|
55
|
+
type: 'textarea',
|
56
|
+
title: 'Location URLs',
|
57
|
+
description: 'A list of URLs for Location files as a JSON-encoded array of strings'
|
58
|
+
|
59
|
+
run do
|
60
|
+
assert locations_json.present?, 'No Location urls received'
|
61
|
+
assert_valid_json(locations_json)
|
62
|
+
profile_url = 'http://fhir-registry.smarthealthit.org/StructureDefinition/vaccine-location'
|
63
|
+
|
64
|
+
location_urls = JSON.parse(locations_json)
|
65
|
+
|
66
|
+
vtrcks_pin_found = false
|
67
|
+
|
68
|
+
location_urls.each do |location_url|
|
69
|
+
validate_response(location_url, profile_url)
|
70
|
+
|
71
|
+
request.response_body&.each_line do |line|
|
72
|
+
break if vtrcks_pin_found
|
73
|
+
|
74
|
+
location = FHIR.from_contents(line)
|
75
|
+
vtrcks_pin_found =
|
76
|
+
location
|
77
|
+
&.identifier
|
78
|
+
&.any? { |identifier| identifier.system == 'https://cdc.gov/vaccines/programs/vtrcks' }
|
79
|
+
|
80
|
+
end
|
81
|
+
|
82
|
+
assert(
|
83
|
+
messages.none? { |message| message[:type] == 'error' },
|
84
|
+
"Succesfully validated #{@valid_resource_count} resources. Test stops after first invalid resource"
|
85
|
+
)
|
86
|
+
end
|
87
|
+
|
88
|
+
assert vtrcks_pin_found, "No Locations included a VTrckS PIN."
|
89
|
+
|
90
|
+
pass "Successfully validated #{@valid_resource_count} resources."
|
91
|
+
end
|
92
|
+
end
|
93
|
+
|
94
|
+
test do
|
95
|
+
include ResourceChecker
|
96
|
+
id :slot_resources
|
97
|
+
title 'Slot Resources'
|
98
|
+
|
99
|
+
input :slots_json,
|
100
|
+
type: 'textarea',
|
101
|
+
title: 'Slot URLs',
|
102
|
+
description: 'A list of URLs for Slot files as a JSON-encoded array of strings'
|
103
|
+
|
104
|
+
run do
|
105
|
+
assert slots_json.present?, 'No Slot urls received'
|
106
|
+
assert_valid_json(slots_json)
|
107
|
+
profile_url = 'http://fhir-registry.smarthealthit.org/StructureDefinition/vaccine-slot'
|
108
|
+
|
109
|
+
slot_urls = JSON.parse(slots_json)
|
110
|
+
|
111
|
+
slot_urls.each do |slot_url|
|
112
|
+
validate_response(slot_url, profile_url)
|
113
|
+
|
114
|
+
assert(
|
115
|
+
messages.none? { |message| message[:type] == 'error' },
|
116
|
+
"Succesfully validated #{@valid_resource_count} resources. Test stops after first invalid resource"
|
117
|
+
)
|
118
|
+
end
|
119
|
+
|
120
|
+
pass "Successfully validated #{@valid_resource_count} resources."
|
121
|
+
end
|
122
|
+
end
|
123
|
+
|
124
|
+
test do
|
125
|
+
include ResourceChecker
|
126
|
+
id :schedule_resources
|
127
|
+
title 'Schedule Resources'
|
128
|
+
|
129
|
+
input :schedules_json,
|
130
|
+
type: 'textarea',
|
131
|
+
title: 'Schedule URLs',
|
132
|
+
description: 'A list of URLs for Schedule files as a JSON-encoded array of strings'
|
133
|
+
|
134
|
+
run do
|
135
|
+
assert schedules_json.present?, 'No Schedule urls received'
|
136
|
+
assert_valid_json(schedules_json)
|
137
|
+
profile_url = 'http://fhir-registry.smarthealthit.org/StructureDefinition/vaccine-schedule'
|
138
|
+
|
139
|
+
schedule_urls = JSON.parse(schedules_json)
|
140
|
+
|
141
|
+
schedule_urls.each do |schedule_url|
|
142
|
+
validate_response(schedule_url, profile_url)
|
143
|
+
|
144
|
+
assert(
|
145
|
+
messages.none? { |message| message[:type] == 'error' },
|
146
|
+
"Succesfully validated #{@valid_resource_count} resources. Test stops after first invalid resource"
|
147
|
+
)
|
148
|
+
end
|
149
|
+
|
150
|
+
pass "Successfully validated #{@valid_resource_count} resources."
|
151
|
+
end
|
152
|
+
end
|
153
|
+
end
|
154
|
+
end
|
@@ -0,0 +1,72 @@
|
|
1
|
+
require_relative 'smart_scheduling_links_test_kit/manifest_group'
|
2
|
+
require_relative 'smart_scheduling_links_test_kit/resource_group'
|
3
|
+
require_relative 'smart_scheduling_links_test_kit/version'
|
4
|
+
|
5
|
+
module SMARTSchedulingLinks
|
6
|
+
class Suite < Inferno::TestSuite
|
7
|
+
id :smart_scheduling_links
|
8
|
+
title 'SMART Scheduling Links'
|
9
|
+
description %(
|
10
|
+
Tests for [SMART Scheduling
|
11
|
+
Links](https://github.com/smart-on-fhir/smart-scheduling-links)'.
|
12
|
+
|
13
|
+
These tests work by retrieving a bulk publication manifest, then
|
14
|
+
retrieving the files listed in the manifest and validating the resources
|
15
|
+
they contain.
|
16
|
+
)
|
17
|
+
version VERSION
|
18
|
+
|
19
|
+
validator do
|
20
|
+
url ENV.fetch('VALIDATOR_URL')
|
21
|
+
exclude_message { |message| message.type == 'info' }
|
22
|
+
perform_additional_validation do |resource, profile_url|
|
23
|
+
next unless profile_url == 'http://fhir-registry.smarthealthit.org/StructureDefinition/vaccine-slot'
|
24
|
+
|
25
|
+
begin
|
26
|
+
start_time = DateTime.parse(resource.start)
|
27
|
+
end_time = DateTime.parse(resource.end)
|
28
|
+
rescue StandardError
|
29
|
+
next
|
30
|
+
end
|
31
|
+
|
32
|
+
messages = []
|
33
|
+
|
34
|
+
slot_hours = (end_time - start_time).days.in_hours
|
35
|
+
|
36
|
+
if slot_hours > 1
|
37
|
+
warning_message = <<~MESSAGE
|
38
|
+
#{resource.resourceType}/#{resource.id}: Slot duration is
|
39
|
+
#{slot_hours.abs.round(1)} hours, but `start` and `end` SHOULD
|
40
|
+
identify a narrow window of time.
|
41
|
+
MESSAGE
|
42
|
+
|
43
|
+
messages << {
|
44
|
+
type: 'warning',
|
45
|
+
message: warning_message
|
46
|
+
}
|
47
|
+
|
48
|
+
has_capacity_extension = resource.extension&.any? do |extension|
|
49
|
+
extension.url == 'http://fhir-registry.smarthealthit.org/StructureDefinition/slot-capacity'
|
50
|
+
end
|
51
|
+
|
52
|
+
if !has_capacity_extension
|
53
|
+
warning_message = <<~MESSAGE
|
54
|
+
#{resource.resourceType}/#{resource.id}: Slot should include a
|
55
|
+
#capacity extension if its duration is longer than an hour.
|
56
|
+
MESSAGE
|
57
|
+
|
58
|
+
messages << {
|
59
|
+
type: 'warning',
|
60
|
+
message: warning_message
|
61
|
+
}
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
messages
|
66
|
+
end
|
67
|
+
end
|
68
|
+
|
69
|
+
group from: :smart_scheduling_links_manifest
|
70
|
+
group from: :smart_scheduling_links_resources
|
71
|
+
end
|
72
|
+
end
|
metadata
ADDED
@@ -0,0 +1,120 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: smart_scheduling_links_test_kit
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.1.0
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Inferno Team
|
8
|
+
autorequire:
|
9
|
+
bindir: bin
|
10
|
+
cert_chain: []
|
11
|
+
date: 2023-02-22 00:00:00.000000000 Z
|
12
|
+
dependencies:
|
13
|
+
- !ruby/object:Gem::Dependency
|
14
|
+
name: inferno_core
|
15
|
+
requirement: !ruby/object:Gem::Requirement
|
16
|
+
requirements:
|
17
|
+
- - "~>"
|
18
|
+
- !ruby/object:Gem::Version
|
19
|
+
version: 0.4.4
|
20
|
+
type: :runtime
|
21
|
+
prerelease: false
|
22
|
+
version_requirements: !ruby/object:Gem::Requirement
|
23
|
+
requirements:
|
24
|
+
- - "~>"
|
25
|
+
- !ruby/object:Gem::Version
|
26
|
+
version: 0.4.4
|
27
|
+
- !ruby/object:Gem::Dependency
|
28
|
+
name: database_cleaner-sequel
|
29
|
+
requirement: !ruby/object:Gem::Requirement
|
30
|
+
requirements:
|
31
|
+
- - "~>"
|
32
|
+
- !ruby/object:Gem::Version
|
33
|
+
version: '1.8'
|
34
|
+
type: :development
|
35
|
+
prerelease: false
|
36
|
+
version_requirements: !ruby/object:Gem::Requirement
|
37
|
+
requirements:
|
38
|
+
- - "~>"
|
39
|
+
- !ruby/object:Gem::Version
|
40
|
+
version: '1.8'
|
41
|
+
- !ruby/object:Gem::Dependency
|
42
|
+
name: factory_bot
|
43
|
+
requirement: !ruby/object:Gem::Requirement
|
44
|
+
requirements:
|
45
|
+
- - "~>"
|
46
|
+
- !ruby/object:Gem::Version
|
47
|
+
version: '6.1'
|
48
|
+
type: :development
|
49
|
+
prerelease: false
|
50
|
+
version_requirements: !ruby/object:Gem::Requirement
|
51
|
+
requirements:
|
52
|
+
- - "~>"
|
53
|
+
- !ruby/object:Gem::Version
|
54
|
+
version: '6.1'
|
55
|
+
- !ruby/object:Gem::Dependency
|
56
|
+
name: rspec
|
57
|
+
requirement: !ruby/object:Gem::Requirement
|
58
|
+
requirements:
|
59
|
+
- - "~>"
|
60
|
+
- !ruby/object:Gem::Version
|
61
|
+
version: '3.10'
|
62
|
+
type: :development
|
63
|
+
prerelease: false
|
64
|
+
version_requirements: !ruby/object:Gem::Requirement
|
65
|
+
requirements:
|
66
|
+
- - "~>"
|
67
|
+
- !ruby/object:Gem::Version
|
68
|
+
version: '3.10'
|
69
|
+
- !ruby/object:Gem::Dependency
|
70
|
+
name: webmock
|
71
|
+
requirement: !ruby/object:Gem::Requirement
|
72
|
+
requirements:
|
73
|
+
- - "~>"
|
74
|
+
- !ruby/object:Gem::Version
|
75
|
+
version: '3.11'
|
76
|
+
type: :development
|
77
|
+
prerelease: false
|
78
|
+
version_requirements: !ruby/object:Gem::Requirement
|
79
|
+
requirements:
|
80
|
+
- - "~>"
|
81
|
+
- !ruby/object:Gem::Version
|
82
|
+
version: '3.11'
|
83
|
+
description: SMART Scheduling Links Test Kit
|
84
|
+
email:
|
85
|
+
- inferno@groups.mitre.org
|
86
|
+
executables: []
|
87
|
+
extensions: []
|
88
|
+
extra_rdoc_files: []
|
89
|
+
files:
|
90
|
+
- LICENSE
|
91
|
+
- lib/smart_scheduling_links_test_kit.rb
|
92
|
+
- lib/smart_scheduling_links_test_kit/manifest_group.rb
|
93
|
+
- lib/smart_scheduling_links_test_kit/resource_group.rb
|
94
|
+
- lib/smart_scheduling_links_test_kit/version.rb
|
95
|
+
homepage: https://github.com/inferno-framework/smart-scheduling-links-test-kit
|
96
|
+
licenses:
|
97
|
+
- Apache-2.0
|
98
|
+
metadata:
|
99
|
+
homepage_uri: https://github.com/inferno-framework/smart-scheduling-links-test-kit
|
100
|
+
source_code_uri: https://github.com/inferno-framework/smart-scheduling-links-test-kit
|
101
|
+
post_install_message:
|
102
|
+
rdoc_options: []
|
103
|
+
require_paths:
|
104
|
+
- lib
|
105
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
106
|
+
requirements:
|
107
|
+
- - ">="
|
108
|
+
- !ruby/object:Gem::Version
|
109
|
+
version: 3.1.2
|
110
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
111
|
+
requirements:
|
112
|
+
- - ">="
|
113
|
+
- !ruby/object:Gem::Version
|
114
|
+
version: '0'
|
115
|
+
requirements: []
|
116
|
+
rubygems_version: 3.3.7
|
117
|
+
signing_key:
|
118
|
+
specification_version: 4
|
119
|
+
summary: SMART Scheduling Links Test Kit
|
120
|
+
test_files: []
|