Skip to content

Commit 38fd9e9

Browse files
committed
[WIP] Fix Delivering too Many Search Results in Some Cases
Closes: #1685
1 parent 6ae041b commit 38fd9e9

File tree

109 files changed

+4595
-548
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

109 files changed

+4595
-548
lines changed

dev/blaze/dev/rocksdb.clj

+1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@
2121
(ac/supply-async #(rocksdb/compact-range! (index-kv-store) :resource-as-of-index))
2222

2323
(doseq [index [:search-param-value-index
24+
:type-search-param-token-full-resource-index
2425
:resource-value-index
2526
:compartment-search-param-value-index
2627
:compartment-resource-type-index

docs/implementation/database.md

+130-9
Original file line numberDiff line numberDiff line change
@@ -120,14 +120,27 @@ The `SystemStats` index keeps track of the total number of resources, and the nu
120120

121121
The indices not depending on `t` directly point to the resource versions by their content hash.
122122

123-
| Name | Key Parts | Value |
124-
|-------------------------------------|----------------------------------------------------------------|-------|
125-
| SearchParamValueResource | search-param, type, value, id, hash-prefix | - |
126-
| ResourceSearchParamValue | type, id, hash-prefix, search-param, value | - |
127-
| CompartmentSearchParamValueResource | comp-code, comp-id, search-param, type, value, id, hash-prefix | - |
128-
| CompartmentResourceType | comp-code, comp-id, type, id | - |
129-
| SearchParam | code, type | id |
130-
| ActiveSearchParams | id | - |
123+
| Name | Key Parts | Value | Since |
124+
|-------------------------------------------|----------------------------------------------------------------|-------|------:|
125+
| SearchParamValueResource | search-param, type, value, id, hash-prefix | - |
126+
| ResourceSearchParamValue | type, id, hash-prefix, search-param, value | - |
127+
| CompartmentSearchParamValueResource | comp-code, comp-id, search-param, type, value, id, hash-prefix | - |
128+
| CompartmentResourceType | comp-code, comp-id, type, id | - |
129+
| TypeSearchParamTokenFullResource | search-param, type, value, system, id, hash-prefix | - | 0.27 |
130+
| TypeSearchParamTokenSystemResource | search-param, type, system, id, hash-prefix | - | 0.27 |
131+
| TypeSearchParamReferenceCanonicalResource | search-param, type, url, version, id, hash-prefix | - | 0.27 |
132+
| TypeSearchParamReferenceUrlResource | search-param, type, url, id, hash-prefix | - | 0.27 |
133+
| TypeSearchParamReferenceLocalResource | search-param, type, ref-id, ref-type, id, hash-prefix | - | 0.27 |
134+
| ResourceSearchParamTokenFull | type, id, hash-prefix, search-param, value, system | - | 0.27 |
135+
| ResourceSearchParamTokenSystem | type, id, hash-prefix, search-param, system | - | 0.27 |
136+
| ResourceSearchParamReferenceCanonical | type, id, hash-prefix, search-param, url, version | - | 0.27 |
137+
| ResourceSearchParamReferenceUrl | type, id, hash-prefix, search-param, url | - | 0.27 |
138+
| ResourceSearchParamReferenceLocal | type, id, hash-prefix, search-param, ref-id, ref-type | - | 0.27 |
139+
| PatientTypeSearchParamTokenFullResource | patient-id, search-param, type, value, system, id, hash-prefix | - | 0.27 |
140+
| SearchParam | code, type | id |
141+
| ActiveSearchParams | id | - |
142+
| SearchParamCode | code | id |
143+
| System | code | id |
131144

132145
#### SearchParamValueResource
133146

@@ -137,7 +150,7 @@ The `SearchParamValueResource` index is used to find resources based on search p
137150
* `type` - a 4-byte hash of the resource type
138151
* `value` - the encoded value of the resource reachable by the search parameters FHIRPath expression. The encoding depends on the search parameters type.
139152
* `id` - the logical id of the resource
140-
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
153+
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
141154

142155
The way the `SearchParamValueResource` index is used, depends on the type of the search parameter. The following sections will explain this in detail for each type:
143156

@@ -223,6 +236,102 @@ The `ResourceSearchParamValue` index is used to decide whether a resource contai
223236
* `search-param` - a 4-byte hash of the search parameters code used to identify the search parameter
224237
* `value` - the encoded value of the resource reachable by the search parameters FHIRPath expression. The encoding depends on the search parameters type.
225238

239+
#### TypeSearchParamTokenFullResource
240+
241+
New index in v0.27.0. It is used to find resources based on full values of search parameters of type token. Full values consist of the system and value for Identifiers or code for Codings. The system will be the special value 0x000000 if not available in the resource.
242+
243+
* `type` - the type byte of the resource type (one byte)
244+
* `search-param` - a 3-byte identifier of the search parameters code used to identify the search parameter
245+
* `value` - the full code/value
246+
* `system` - a 3-byte identifier of the system URI
247+
* `id` - the logical id of the resource
248+
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
249+
250+
#### TypeSearchParamTokenSystemResource
251+
252+
New index in v0.27.0. It is used to find resources based on the system only of search parameters of type token. If the system is not available, no index entry will be written.
253+
254+
* `type` - the type byte of the resource type (one byte)
255+
* `search-param` - a 3-byte identifier of the search parameters code used to identify the search parameter
256+
* `system` - a 3-byte identifier of the system URI
257+
* `id` - the logical id of the resource
258+
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
259+
260+
#### TypeSearchParamReferenceCanonicalResource
261+
262+
New index in v0.27.0. It is used to find resources based on the reference value in case it is an canonical URL of search parameters of type reference.
263+
264+
* `type` - the type byte of the resource type (one byte)
265+
* `search-param` - a 3-byte identifier of the search parameters code used to identify the search parameter
266+
* `url` - a 4-byte identifier of the canonical URL
267+
* `version` - the full version
268+
* `id` - the logical id of the resource
269+
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
270+
271+
#### TypeSearchParamReferenceUrlResource
272+
273+
New index in v0.27.0. It is used to find resources based on the reference value in case it is an URL of search parameters of type reference.
274+
275+
* `type` - the type byte of the resource type (one byte)
276+
* `search-param` - a 3-byte identifier of the search parameters code used to identify the search parameter
277+
* `url` - the full url
278+
* `id` - the logical id of the resource
279+
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
280+
281+
#### TypeSearchParamReferenceLocalResource
282+
283+
New index in v0.27.0. It is used to find resources based on the reference value in case it is a local reference of search parameters of type reference.
284+
285+
* `type` - the type byte of the resource type (one byte)
286+
* `search-param` - a 3-byte identifier of the search parameters code used to identify the search parameter
287+
* `ref-id` - the logical id of the referenced resource
288+
* `ref-type` - the type byte of the referenced resource type (one byte)
289+
* `id` - the logical id of the resource
290+
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
291+
292+
#### ResourceSearchParamTokenFull
293+
294+
* `type` - the type byte of the resource type (one byte)
295+
* `id` - the logical id of the resource
296+
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
297+
* `search-param` - a 3-byte identifier of the search parameters code used to identify the search parameter
298+
* `value` - the full code/value
299+
* `system` - a 3-byte identifier of the system URI
300+
301+
#### ResourceSearchParamTokenSystem
302+
303+
* `type` - the type byte of the resource type (one byte)
304+
* `id` - the logical id of the resource
305+
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
306+
* `search-param` - a 3-byte identifier of the search parameters code used to identify the search parameter
307+
* `system` - a 3-byte identifier of the system URI
308+
309+
#### ResourceSearchParamReferenceCanonical
310+
311+
* `type` - the type byte of the resource type (one byte)
312+
* `id` - the logical id of the resource
313+
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
314+
* `search-param` - a 3-byte identifier of the search parameters code used to identify the search parameter
315+
* `url` - a 4-byte identifier of the canonical URL
316+
* `version` - the full version
317+
318+
#### ResourceSearchParamReferenceUrl
319+
320+
* `type` - the type byte of the resource type (one byte)
321+
* `id` - the logical id of the resource
322+
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
323+
* `search-param` - a 3-byte identifier of the search parameters code used to identify the search parameter
324+
* `url` - the full url
325+
326+
#### ResourceSearchParamReferenceLocal
327+
328+
* `type` - the type byte of the resource type (one byte)
329+
* `id` - the logical id of the resource
330+
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
331+
* `search-param` - a 3-byte identifier of the search parameters code used to identify the search parameter
332+
* `ref-id` - the logical id of the referenced resource
333+
* `ref-type` - the type byte of the referenced resource type (one byte)
334+
226335
#### CompartmentSearchParamValueResource
227336

228337
The `CompartmentSearchParamValueResource` index is used to find resources of a particular compartment based on search parameter values.
@@ -236,6 +345,18 @@ The `CompartmentResourceType` index is used to find all resources that belong to
236345
* `type` - a 4-byte hash of the resource type of the resource that belongs to the compartment, ex. `Observation`
237346
* `id` - the logical id of the resource that belongs to the compartment, ex. the logical id of the Observation
238347

348+
#### PatientTypeSearchParamTokenFullResource
349+
350+
New index in v0.27.0. It is used to find resources based on full values of search parameters of type token. Full values consist of the system and value for Identifiers or code for Codings. The system will be the special value 0x000000 if not available in the resource.
351+
352+
* `patient-id` - the logical id of the patient
353+
* `type` - the type byte of the resource type (one byte)
354+
* `search-param` - a 3-byte identifier of the search parameters code used to identify the search parameter
355+
* `value` - the full code/value
356+
* `system` - a 3-byte identifier of the system URI
357+
* `id` - the logical id of the resource
358+
* `hash-prefix` - a 4-byte prefix of the content-hash of the resource version
359+
239360
#### ActiveSearchParams
240361

241362
Currently not used.

modules/admin-api/test/blaze/admin_api_test.clj

+7-35
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
[blaze.admin-api :as admin-api]
44
[blaze.async.comp :as ac :refer [do-sync]]
55
[blaze.db.api :as d]
6-
[blaze.db.api-stub]
6+
[blaze.db.api-stub :as api-stub]
77
[blaze.db.impl.index.patient-last-change :as plc]
88
[blaze.db.kv :as-alias kv]
99
[blaze.db.kv.rocksdb :as rocksdb]
@@ -97,50 +97,21 @@
9797
{:dir (str dir "/index")
9898
:block-cache (ig/ref ::rocksdb/block-cache)
9999
:column-families
100-
{:search-param-value-index
101-
{:write-buffer-size-in-mb 1
102-
:max-write-buffer-number 1
103-
:max-bytes-for-level-base-in-mb 1
104-
:target-file-size-base-in-mb 1}
105-
:resource-value-index nil
100+
(assoc
101+
api-stub/index-kv-store-column-families
106102
:compartment-search-param-value-index
107103
{:write-buffer-size-in-mb 1
108104
:max-write-buffer-number 1
109105
:max-bytes-for-level-base-in-mb 1
110106
:target-file-size-base-in-mb 1}
111-
:compartment-resource-type-index nil
112-
:active-search-params nil
113-
:tx-success-index {:reverse-comparator? true}
114-
:tx-error-index nil
115-
:t-by-instant-index {:reverse-comparator? true}
116-
:resource-as-of-index nil
117-
:type-as-of-index nil
118-
:system-as-of-index nil
119107
:patient-last-change-index
120108
{:write-buffer-size-in-mb 1
121109
:max-write-buffer-number 1
122110
:max-bytes-for-level-base-in-mb 1
123-
:target-file-size-base-in-mb 1}
124-
:type-stats-index nil
125-
:system-stats-index nil
126-
:cql-bloom-filter nil
127-
:cql-bloom-filter-by-t nil}}
111+
:target-file-size-base-in-mb 1})}
128112

129113
[::kv/mem :blaze.db.admin/index-kv-store]
130-
{:column-families
131-
{:search-param-value-index nil
132-
:resource-value-index nil
133-
:compartment-search-param-value-index nil
134-
:compartment-resource-type-index nil
135-
:active-search-params nil
136-
:tx-success-index {:reverse-comparator? true}
137-
:tx-error-index nil
138-
:t-by-instant-index {:reverse-comparator? true}
139-
:resource-as-of-index nil
140-
:type-as-of-index nil
141-
:system-as-of-index nil
142-
:type-stats-index nil
143-
:system-stats-index nil}}
114+
{:column-families api-stub/index-kv-store-column-families}
144115

145116
::rs/kv
146117
{:kv-store (ig/ref :blaze.db/resource-kv-store)
@@ -169,7 +140,8 @@
169140
[:blaze.db.node.resource-indexer/executor :blaze.db.node.resource-indexer.admin/executor] {}
170141

171142
:blaze.db/search-param-registry
172-
{:structure-definition-repo structure-definition-repo}
143+
{:kv-store (ig/ref :blaze.db.main/index-kv-store)
144+
:structure-definition-repo structure-definition-repo}
173145

174146
::rocksdb/block-cache {:size-in-mb 1}
175147

modules/byte-buffer/src/blaze/byte_buffer.clj

+23-8
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,12 @@
9191
(.copyTo ^ByteString byte-string byte-buffer)
9292
byte-buffer)
9393

94+
(defn put-null-terminated-byte-string!
95+
"Copies all bytes of `byte-string` into `byte-buffer`."
96+
[byte-buffer byte-string]
97+
(.copyTo ^ByteString byte-string byte-buffer)
98+
(put-byte! byte-buffer 0))
99+
94100
(defn limit
95101
"Returns the limit of `byte-buffer`."
96102
{:inline
@@ -120,6 +126,13 @@
120126
[byte-buffer position]
121127
(.position ^ByteBuffer byte-buffer (int position)))
122128

129+
(defn inc-position!
130+
{:inline
131+
(fn [byte-buffer amount]
132+
`(set-position! ~byte-buffer (unchecked-add-int (position ~byte-buffer) (int ~amount))))}
133+
[byte-buffer amount]
134+
(set-position! byte-buffer (+ (position byte-buffer) (long amount))))
135+
123136
(defn remaining
124137
"Returns the number of elements between the current position and the limit."
125138
{:inline
@@ -229,19 +242,21 @@
229242
[byte-buffer]
230243
(when (pos? (remaining byte-buffer))
231244
(mark! byte-buffer)
232-
(loop [byte (bit-and (long (get-byte! byte-buffer)) 0xFF)
245+
(loop [byte (long (get-byte! byte-buffer))
233246
size 0]
234-
(cond
235-
(zero? byte)
247+
(if (zero? byte)
236248
(do (reset! byte-buffer)
237249
size)
238250

239-
(pos? (remaining byte-buffer))
240-
(recur (bit-and (long (get-byte! byte-buffer)) 0xFF) (inc size))
251+
(if (zero? (remaining byte-buffer))
252+
(do (reset! byte-buffer)
253+
nil)
254+
(recur (long (get-byte! byte-buffer)) (inc size)))))))
241255

242-
:else
243-
(do (reset! byte-buffer)
244-
nil)))))
256+
(defn skip-null-terminated! [byte-buffer]
257+
(if-let [size (size-up-to-null byte-buffer)]
258+
(set-position! byte-buffer (+ (position byte-buffer) (long size) 1))
259+
(throw (Exception. "Can't skip null terminated byte sequence."))))
245260

246261
(defn mismatch
247262
"Finds and returns the relative index of the first mismatch between `a` and

modules/byte-buffer/src/blaze/byte_buffer_spec.clj

+31-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,8 @@
11
(ns blaze.byte-buffer-spec
22
(:require
33
[blaze.byte-buffer :as bb :refer [byte-buffer?]]
4-
[clojure.spec.alpha :as s]))
4+
[clojure.spec.alpha :as s])
5+
(:import [com.google.protobuf ByteString]))
56

67
(s/fdef bb/allocate
78
:args (s/cat :capacity nat-int?)
@@ -31,10 +32,39 @@
3132
:args (s/cat :byte-buffer byte-buffer? :x int?)
3233
:ret byte-buffer?)
3334

35+
(s/fdef bb/put-byte-array!
36+
:args (s/cat :byte-buffer byte-buffer? :byte-array bytes?
37+
:offset (s/? nat-int?) :length (s/? nat-int?))
38+
:ret byte-buffer?)
39+
40+
(s/fdef bb/put-byte-buffer!
41+
:args (s/cat :dst byte-buffer? :src byte-buffer?)
42+
:ret byte-buffer?)
43+
44+
(s/fdef bb/put-byte-string!
45+
:args (s/cat :byte-buffer byte-buffer? :byte-string #(instance? ByteString %))
46+
:ret byte-buffer?)
47+
48+
(s/fdef bb/put-null-terminated-byte-string!
49+
:args (s/cat :byte-buffer byte-buffer? :byte-string #(instance? ByteString %))
50+
:ret byte-buffer?)
51+
3452
(s/fdef bb/limit
3553
:args (s/cat :byte-buffer byte-buffer?)
3654
:ret nat-int?)
3755

56+
(s/fdef bb/position
57+
:args (s/cat :byte-buffer byte-buffer?)
58+
:ret nat-int?)
59+
60+
(s/fdef bb/set-position!
61+
:args (s/cat :byte-buffer byte-buffer? :position nat-int?)
62+
:ret byte-buffer?)
63+
3864
(s/fdef bb/size-up-to-null
3965
:args (s/cat :byte-buffer byte-buffer?)
4066
:ret (s/nilable nat-int?))
67+
68+
(s/fdef bb/skip-null-terminated!
69+
:args (s/cat :byte-buffer byte-buffer?)
70+
:ret byte-buffer?)

modules/byte-string/src/blaze/byte_string.clj

+5-4
Original file line numberDiff line numberDiff line change
@@ -64,14 +64,15 @@
6464

6565
(defn from-byte-buffer-null-terminated!
6666
"Returns the bytes from `byte-buffer` up to (exclusive) a null byte (0x00) as
67-
byte string ot nil if `byte-buffer` doesn't include a null byte.
67+
byte string or nil if `byte-buffer` doesn't include a null byte.
6868
6969
Increments the position of `byte-buffer` up to including the null byte."
7070
[byte-buffer]
71-
(when-let [size (bb/size-up-to-null byte-buffer)]
71+
(if-let [size (bb/size-up-to-null byte-buffer)]
7272
(let [bs (from-byte-buffer! byte-buffer size)]
73-
(bb/get-byte! byte-buffer)
74-
bs)))
73+
(bb/set-position! byte-buffer (inc (bb/position byte-buffer)))
74+
bs)
75+
(throw (Exception. "Can't read null terminated byte string."))))
7576

7677
(defn from-hex [s]
7778
(ByteString/copyFrom (.decode (BaseEncoding/base16) s)))

modules/byte-string/src/blaze/byte_string_spec.clj

+1-1
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@
2727

2828
(s/fdef bs/from-byte-buffer-null-terminated!
2929
:args (s/cat :byte-buffer byte-buffer?)
30-
:ret (s/nilable bs/byte-string?))
30+
:ret bs/byte-string?)
3131

3232
(s/fdef bs/from-hex
3333
:args (s/cat :s string?)

0 commit comments

Comments
 (0)